Slide 1

Slide 1 text

Working e!ectively at scale

Slide 2

Slide 2 text

Francisco Díaz franciscodiaz.cl - @fco_diaz

Slide 3

Slide 3 text

Startups 2011 - 2017

Slide 4

Slide 4 text

Airbnb 2017 - today

Slide 5

Slide 5 text

No content

Slide 6

Slide 6 text

No content

Slide 7

Slide 7 text

No content

Slide 8

Slide 8 text

No content

Slide 9

Slide 9 text

No content

Slide 10

Slide 10 text

No content

Slide 11

Slide 11 text

No content

Slide 12

Slide 12 text

No content

Slide 13

Slide 13 text

No content

Slide 14

Slide 14 text

organizations ... are constrained to produce designs which are copies of the communication structures of these organizations — Conway's law

Slide 15

Slide 15 text

How do you divide your codebase?

Slide 16

Slide 16 text

Architectural layer

Slide 17

Slide 17 text

User Flow

Slide 18

Slide 18 text

What about Airbnb?

Slide 19

Slide 19 text

No content

Slide 20

Slide 20 text

1 million lines of Swift

Slide 21

Slide 21 text

~80 commits to master on any given day to the repo (Android + iOS)

Slide 22

Slide 22 text

Bigger buckets

Slide 23

Slide 23 text

No content

Slide 24

Slide 24 text

A user should be able to wishlist a listing from the booking flow

Slide 25

Slide 25 text

How do they relate with each other?

Slide 26

Slide 26 text

No content

Slide 27

Slide 27 text

No content

Slide 28

Slide 28 text

No content

Slide 29

Slide 29 text

50 min local clean builds

Slide 30

Slide 30 text

~30 min !

Slide 31

Slide 31 text

Buck HTTP Cache https://github.com/airbnb/BuckSample

Slide 32

Slide 32 text

No content

Slide 33

Slide 33 text

~5 min !

Slide 34

Slide 34 text

No content

Slide 35

Slide 35 text

No content

Slide 36

Slide 36 text

Dependency inversion

Slide 37

Slide 37 text

‣High-level modules should not depend on low- level modules. Both should depend on abstractions. ‣Abstractions should not depend on details. Details (concrete implementations) should depend on abstractions.

Slide 38

Slide 38 text

No content

Slide 39

Slide 39 text

A user should be able to wishlist a listing from the booking flow

Slide 40

Slide 40 text

No content

Slide 41

Slide 41 text

Easy! WishListDataSource + interface!

Slide 42

Slide 42 text

No content

Slide 43

Slide 43 text

WishListDataSource is still visible

Slide 44

Slide 44 text

No content

Slide 45

Slide 45 text

Socializing best practices

Slide 46

Slide 46 text

!

Slide 47

Slide 47 text

+60 iOS developers

Slide 48

Slide 48 text

Automating best practices

Slide 49

Slide 49 text

Groups Modules

Slide 50

Slide 50 text

Groups Modules Module Types

Slide 51

Slide 51 text

Module Types Feature + Interface Service + Interface

Slide 52

Slide 52 text

Feature A screen or a flow in the app

Slide 53

Slide 53 text

Service Manage shared state or resources

Slide 54

Slide 54 text

No content

Slide 55

Slide 55 text

No content

Slide 56

Slide 56 text

How do we enforce these best practices?

Slide 57

Slide 57 text

/services /service_interfaces /features /feature_interfaces

Slide 58

Slide 58 text

def service_interface( name, deps): max_visibility = [ "//ios/feature_interfaces/...", "//ios/features/...", "//ios/service_interfaces/...", "//ios/services/...", ]

Slide 59

Slide 59 text

service_interface( name = "Networking", deps = [ "//ios/service_interfaces/Logging", ], )

Slide 60

Slide 60 text

feature( name = "Booking", deps = [ "//ios/service_interfaces/Networking", "//ios/service_interfaces/WishListService", ], )

Slide 61

Slide 61 text

iOS Platform

Slide 62

Slide 62 text

Module creation needs to be easy

Slide 63

Slide 63 text

No content

Slide 64

Slide 64 text

No content

Slide 65

Slide 65 text

rake make:module

Slide 66

Slide 66 text

> Provide the type of module you want to create: 1: Non Platform 2: Feature 3: Feature Interface 4: Service 5: Service Interface 4 > New module name: Swiftable > Provide a high level description of this module: This is a module to present at Swiftable

Slide 67

Slide 67 text

Buck Human readable dependencies https://github.com/airbnb/BuckSample

Slide 68

Slide 68 text

feature( name = "Booking", deps = [ "//ios/service_interfaces/Networking", "//ios/service_interfaces/WishListService", "//ios/feature_interfaces/HelpCenter", ], )

Slide 69

Slide 69 text

No content

Slide 70

Slide 70 text

Feature A screen or a flow in the app

Slide 71

Slide 71 text

No content

Slide 72

Slide 72 text

No content

Slide 73

Slide 73 text

No content

Slide 74

Slide 74 text

Dev Apps

Slide 75

Slide 75 text

~1 min Dev Apps

Slide 76

Slide 76 text

Big buckets Small playgrounds

Slide 77

Slide 77 text

No content

Slide 78

Slide 78 text

How do we get there?

Slide 79

Slide 79 text

~100 modules One module type: /libraries

Slide 80

Slide 80 text

libraries/AirbnbBooking libraries/AirbnbBusinessTravel libraries/AirbnbHelpCenter libraries/AirbnbListings libraries/AirbnbNetworking libraries/AirbnbWishLists ...

Slide 81

Slide 81 text

Before iOS Platform /libraries

Slide 82

Slide 82 text

On the iOS Platform /services /service_interfaces /features /feature_interfaces

Slide 83

Slide 83 text

How to get everybody on the iOS Platform?

Slide 84

Slide 84 text

Remove libraries/ and start over

Slide 85

Slide 85 text

!

Slide 86

Slide 86 text

Progressively migrate

Slide 87

Slide 87 text

Let's migrate WishLists Data Source

Slide 88

Slide 88 text

libraries/WishLists

Slide 89

Slide 89 text

No content

Slide 90

Slide 90 text

No content

Slide 91

Slide 91 text

No content

Slide 92

Slide 92 text

What are the dependency rules for libraries/?

Slide 93

Slide 93 text

No content

Slide 94

Slide 94 text

Inbound dependencies Outbound dependencies

Slide 95

Slide 95 text

Inbound dependencies Outbound dependencies

Slide 96

Slide 96 text

No content

Slide 97

Slide 97 text

The interface module has stricter rules

Slide 98

Slide 98 text

Migrate all the call sites

Slide 99

Slide 99 text

As the owner of WishLists We don't control these

Slide 100

Slide 100 text

No content

Slide 101

Slide 101 text

Calculated tech debt

Slide 102

Slide 102 text

Inbound dependencies Outbound dependencies

Slide 103

Slide 103 text

No content

Slide 104

Slide 104 text

we know our usage of Networking

Slide 105

Slide 105 text

We control our dependencies

Slide 106

Slide 106 text

Allow inbound dependencies from libraries/ Don't allow outbound dependencies to libraries/

Slide 107

Slide 107 text

libraries/ has access to the iOS Platform

Slide 108

Slide 108 text

The iOS Platform doesn't have access to libraries/

Slide 109

Slide 109 text

Code on the iOS Platform has good boundaries

Slide 110

Slide 110 text

while we allow for easier migration

Slide 111

Slide 111 text

def service_interface( name, visibility = []): max_visibility = [ "//ios/feature_interfaces/...", "//ios/features/...", "//ios/service_interfaces/...", "//ios/services/...", ] add_visibility_for_legacy_module_structure(max_visibility)

Slide 112

Slide 112 text

def add_visibility_for_legacy_module_structure(visibility): visibility.extend([ "//ios/libraries/...", ])

Slide 113

Slide 113 text

We started migrating from the bottom up

Slide 114

Slide 114 text

Try it ourselves !rst

Slide 115

Slide 115 text

Pilot with others teams

Slide 116

Slide 116 text

Should I implement this?

Slide 117

Slide 117 text

Most likely NO

Slide 118

Slide 118 text

There's no silver bullet

Slide 119

Slide 119 text

Adapt

Slide 120

Slide 120 text

Summary 1. Figure out where you're struggling 2.Create and document best practices 3.Automate best practices where needed

Slide 121

Slide 121 text

¡Gracias! franciscodiaz.cl/talks