Upgrade to Pro — share decks privately, control downloads, hide ads and more …

Mali Akmanalp - Library UX: Using abstraction t...

Mali Akmanalp - Library UX: Using abstraction towards friendlier APIs

Complicated libraries can be a pain in the butt to use. It's not surprising that there are a lot of "X for humans" libraries out there, some of which are mostly wrappers around more frustrating interfaces.

This is not a theoretical talk. I'll touch upon theory to give you context, but will then talk about what that means for you in practice so that you can write better libraries. I'll talk about why library UX matters, about abstraction as a general concept, about out what happens when you over/under abstract, and about some useful tips to help build friendly APIs. Meanwhile, I'll show some positive examples from libraries we know and love (flask, SQLAlchemy, Requests, etc). Once you recognize these effects in play, you'll be able to apply them to your own code and make life better for everyone!

https://us.pycon.org/2017/schedule/presentation/781/

PyCon 2017

May 21, 2017
Tweet

More Decks by PyCon 2017

Other Decks in Programming

Transcript

  1. Because you’re a nice person Library UX Using abstraction towards

    friendlier APIs Mali Akmanalp (@makmanalp) hear at the back Who here has had to write code that other people have to consume?
  2. UX == User Experience == How this makes me feel

    How do I feel when I use this thing? excited / frustrated product companies think about UX - e.g. apple products
  3. UX == User Experience == Usability overlapping How easy is

    it to learn and use this thing? — both come late into consideration, if ever talk by kenneth reitz
  4. import urllib2 gh_url = 'https://api.github.com/user' req = urllib2.Request(gh_url) password_manager =

    urllib2.HTTPPasswordMgrWithDefaultRealm() password_manager.add_password(None, gh_url, 'user', 'pass') auth_manager = urllib2.HTTPBasicAuthHandler(password_manager) opener = urllib2.build_opener(auth_manager) urllib2.install_opener(opener) handler = urllib2.urlopen(req) print handler.read() https://www.kennethreitz.org/python-for-humans/ github API HTTP request custom class
  5. import requests url = 'https://api.github.com/user' auth = ('username', 'password') r

    = requests.get(url, auth=auth) print r.content https://www.kennethreitz.org/python-for-humans/ Drastically different experience “urllib is so terrible” || mad at me Don't jump to conclusions / Hold that thought - Alternate theory
  6. import urllib2 gh_url = 'https://api.github.com/user' req = urllib2.Request(gh_url) password_manager =

    urllib2.HTTPPasswordMgrWithDefaultRealm() password_manager.add_password(None, gh_url, 'user', 'pass') auth_manager = urllib2.HTTPBasicAuthHandler(password_manager) opener = urllib2.build_opener(auth_manager) urllib2.install_opener(opener) handler = urllib2.urlopen(req) print handler.read() What is each parameter? noun_verb or verbnoun? reduces mistakes https://www.ncbi.nlm.nih.gov/pmc/articles/PMC4173163/
  7. import urllib2 gh_url = 'https://api.github.com/user' req = urllib2.Request(gh_url) password_manager =

    urllib2.HTTPPasswordMgrWithDefaultRealm() password_manager.add_password(None, gh_url, 'user', 'pass') auth_manager = urllib2.HTTPBasicAuthHandler(password_manager) opener = urllib2.build_opener(auth_manager) urllib2.install_opener(opener) handler = urllib2.urlopen(req) print handler.read() What is each parameter? noun_verb or verbnoun? reduces mistakes https://www.ncbi.nlm.nih.gov/pmc/articles/PMC4173163/
  8. import urllib2 gh_url = 'https://api.github.com/user' req = urllib2.Request(gh_url) password_manager =

    urllib2.HTTPPasswordMgrWithDefaultRealm() password_manager.add_password(None, gh_url, 'user', 'pass') auth_manager = urllib2.HTTPBasicAuthHandler(password_manager) opener = urllib2.build_opener(auth_manager) urllib2.install_opener(opener) handler = urllib2.urlopen(req) print handler.read() What is each parameter? noun_verb or verbnoun? reduces mistakes https://www.ncbi.nlm.nih.gov/pmc/articles/PMC4173163/
  9. import urllib2 gh_url = 'https://api.github.com/user' req = urllib2.Request(gh_url) password_manager =

    urllib2.HTTPPasswordMgrWithDefaultRealm() password_manager.add_password(None, gh_url, 'user', 'pass') auth_manager = urllib2.HTTPBasicAuthHandler(password_manager) opener = urllib2.build_opener(auth_manager) urllib2.install_opener(opener) handler = urllib2.urlopen(req) print handler.read() What is each parameter? noun_verb or verbnoun? reduces mistakes https://www.ncbi.nlm.nih.gov/pmc/articles/PMC4173163/
  10. Good UX minimizes distractions. I don’t care about your lib's

    impl details it's not about you - it's about me Your library is a means to my end. Just let me do my job!
  11. Good UX makes complex tasks routine. “batteries included” feeling of

    tools in my toolbox boilerplate song and dance / incantations Requests + beautifulsoup + pandas + seaborn
  12. http://abstrusegoose.com/307 abstraction analyze the joke to death stack of pancakes

    asteroids->empty void of space Abstraction works even if the architects don't know the very blocks they're building upon
  13. Claim: We are primarily in the business of dealing with

    abstractions. we would do well to pay more attention to them
  14. http://abstrusegoose.com/307 You are here Python: fall in the middle theory

    Py aweosmeness: HL enough -> goal-driven needs C n - but what is abs really?
  15. import requests url = 'https://api.github.com/user' auth = ('username', 'password') r

    = requests.get(url, auth=auth) print r.content https://www.kennethreitz.org/python-for-humans/ requests is highly abstracted less code == less errors
  16. Hiding details makes complex tasks routine. Cool thing: I don’t

    need to know maxwell’s eqs. to make a game helps reason about complex ideas
  17. https://xkcd.com/378/ make fun of “real programmer” Butterfly based disk I/O

    most of your time is spent thinking about butterflies
  18. Hiding details provides a stable interface. changes under the hood

    zero-cost for your users testing is easier!
  19. Functions >>> a array([[ 2., 8., 0., 6.], [ 4.,

    5., 1., 1.], [ 8., 9., 3., 6.]]) >>> np.resize(a, (2,6)) array([[ 2., 8., 0., 6., 4., 5.], [ 1., 1., 8., 9., 3., 6.]]) functions abstract away the details edge cases blasphemies: speed
  20. Classes class User(Base): __tablename__ = 'users' id = Column(Integer, primary_key=True)

    name = Column(String(50)) dessert = Column(String(50)) another tool sqlalchemy models magic
  21. Classes >>> mali.name "mali" >>> mali.name = "mali2" >>> session.add(mali)

    >>> session.commit() classes make state explicit and organized group state and behavior - clean behavior that changes with state
  22. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) n
  23. Leaky Abstractions In [5]: %%timeit ...: for i in range(size):

    ...: for j in range(size): ...: x = big_table[j][i] ...: 10 loops, best of 3: 163 ms per loop
  24. Leaky Abstractions In [5]: %%timeit ...: for i in range(size):

    ...: for j in range(size): ...: x = big_table[i][j] ...: 10 loops, best of 3: 97.1 ms per loop ???
  25. Leaky Abstractions In [5]: %%timeit ...: for i in range(size):

    ...: for j in range(size): ...: x = big_table[i][j] ...: 10 loops, best of 3: 97.1 ms per loop ???
  26. Leaky Abstractions In [5]: %%timeit ...: for i in range(size):

    ...: for j in range(size): ...: x = big_table[i][j] ...: 10 loops, best of 3: 97.1 ms per loop ???
  27. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  28. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  29. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  30. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  31. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  32. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  33. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  34. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  35. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  36. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  37. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  38. Leaky Abstractions [[ 2, 8, …, 0, 6], [ 4,

    5, …, 1, 1], [ …, …, …, …, …], [ …, …, …, …, …], [ 8, 2, …, 5, 6], [ 8, 9, …, 3, 6]]) against the grain "law of leaky abstractions" acceptable compromise
  39. Guts everywhere State everywhere Control flow everywhere Hard to understand

    things Sign: People w/ domain knowledge have a hard time understanding
  40. Cohesion: A thing that does too many things at the

    same time. low-cohesion (contents not related to each other) n
  41. DECIDING ON THE LEVEL OF ABSTRACTION so, you're building a

    shiny new app how to structure abstraction stack
  42. ???

  43. Press release first PM saying: —— next —— "library allows

    developers to compose queries programmatically" "Tired of repetitive CRUD app code? No more!!!" "machine learning in 3 lines of code!" "Data munging? No Problem!" - tears of joy dask.delayed
  44. Press release first PM saying: —— next —— "library allows

    developers to compose queries programmatically" "Tired of repetitive CRUD app code? No more!!!" "machine learning in 3 lines of code!" "Data munging? No Problem!" - tears of joy dask.delayed
  45. Press release first PM saying: —— next —— "library allows

    developers to compose queries programmatically" "Tired of repetitive CRUD app code? No more!!!" "machine learning in 3 lines of code!" "Data munging? No Problem!" - tears of joy dask.delayed
  46. Press release first PM saying: —— next —— "library allows

    developers to compose queries programmatically" "Tired of repetitive CRUD app code? No more!!!" "machine learning in 3 lines of code!" "Data munging? No Problem!" - tears of joy dask.delayed
  47. Press release first PM saying: —— next —— "library allows

    developers to compose queries programmatically" "Tired of repetitive CRUD app code? No more!!!" "machine learning in 3 lines of code!" "Data munging? No Problem!" - tears of joy dask.delayed
  48. "Imaginary Code" second play pretend write code using your imaginary

    library forces you to be specific ~~ contract-first development
  49. Rewrite usage examples with existing libraries how have other people

    tackled similar problems? Is there an improvement?
  50. What does it cost me? pitfalls: cohesion, coupling surface area

    potential cost of changing / un-doing this abstraction?
  51. How likely is this to change? generalizations for events that

    won't happen Do you really need an n-dimensional chess-board class just to make a chess game?
  52. Build less structure up front find out that you were

    off about what you're building don't fix problems before you have them add incrementally avoid "second system effect" Fred Brooks
  53. n

  54. Flask def add_url_rule(self, rule, **options, ...): # ... rule =

    self.url_rule_class(rule, **options, ...) # ... self.url_map.add(rule) # ... https://github.com/pallets/flask/blob/501f0431259a30569a5e62bcce68d102fc3ef993/flask/app.py#L1071 If I look at url_map - werkzeug Map object! Multiple access points “under the hood” - no wall requests / urllib3 obj
  55. Flask def add_url_rule(self, rule, **options, ...): # ... rule =

    self.url_rule_class(rule, **options, ...) # ... self.url_map.add(rule) # ... https://github.com/pallets/flask/blob/501f0431259a30569a5e62bcce68d102fc3ef993/flask/app.py#L1071 If I look at url_map - werkzeug Map object! Multiple access points “under the hood” - no wall requests / urllib3 obj
  56. Flask def add_url_rule(self, rule, **options, ...): # ... rule =

    self.url_rule_class(rule, **options, ...) # ... self.url_map.add(rule) # ... https://github.com/pallets/flask/blob/501f0431259a30569a5e62bcce68d102fc3ef993/flask/app.py#L1071 If I look at url_map - werkzeug Map object! Multiple access points “under the hood” - no wall requests / urllib3 obj
  57. Bokeh Charts Bokeh Glyphs Bokeh JS interactive online visualization create

    exact same chart in all 3 different levels of configurability and effort expose all 3!
  58. Requests vs urllib2 import requests url = 'https://api.github.com/user' auth =

    ('username', 'password') r = requests.get(url, auth=auth) print r.content import urllib2 gh_url = 'https://api.github.com/ user' req = urllib2.Request(gh_url) password_manager = urllib2.HTTPPasswordMgrWithDefaultRea lm() password_manager.add_password(None, gh_url, 'user', 'pass') auth_manager = urllib2.HTTPBasicAuthHandler(password _manager) opener = urllib2.build_opener(auth_manager) urllib2.install_opener(opener) handler = urllib2.urlopen(req) print handler.read() is Urllib2 under-abstracted? audience-specific urllib “wrong” level for you, right level for library devs
  59. Theory: “For Humans” adds a layer we didn’t know we

    were missing. (pause) we were sorely lacking it
  60. Abstraction isn't a goal, It's a tool. can be used

    for good or bad it's everywhere it's a tool we inadvertently use hopefully you start seeing it everywhere / make UX better
  61. Classes """ select * from users where name = '{}';

    """.format("mali") now we need to add another parameter
  62. Classes """ select * from users where name = '{}'

    and dessert = '{}'; """.format("mali", "pie") ugh
  63. Classes class User(Base): __tablename__ = 'users' id = Column(Integer,primary_key=True) name

    = Column(String(50)) dessert = Column(String(50)) classes make state explicit and organized associate and group state and behavior
  64. Read a lot of code! Start with a library you

    use all the time Start with a function you know Find the code and read! Ask for help!
  65. Hiding details papers over grossness??? Cory B speaking about the

    guts of requests https://us.pycon.org/2017/schedule/presentation/71/
  66. https://xkcd.com/378/ make fun of “real programmer” Butterfly vs Magnet needle

    vs File Systems Files aren’t “real” it’s all zeroes and ones! to real apps