Skip to content

Notes of "Software Engineering at Google" by O'Reilly

notes of “Software Engineering at Google” by O’Reilly

Preface

It is important to build not only scalable, BUT also sustainable software, that can adapt to potential changes, while been accessible to other devs

This book will also explore the difference between development AND engineering

  • software is becoming more and more included into our day2day lives, so it is important to engineer it, not just develop and pray
    • thats for other engineers has practices and standards

Overall, here you can find processes, practices and tools, used by Google to achieve healthy codebase, that explored through this topics:

  • effect of time over sustainability
  • effect of scale over viability
  • design trade-offs

When creating software, remember:

  • how code will adapt over time
  • how code will scale with business
  • what trade-offs can be adapted to scale faster OR grow in more sustainable manner

Main topics will be:

  • Google’s culture
  • Processes at scale
  • Tooling (internal AND open-source)

Note, that this is all about the way, how Google operates, not one for all fit list of advices

What is Software Engineering

As software engineers, we have a responsibility to:

  • think about time
  • think about scale
  • make complex decisions

Basically, development is process of writing code, which, if combined with modification, maintenance and team-work, becomes software engineering

  • often you need to decide how long-lived your program will be AND this will result in what will you do(program or engineer)
  • if you can’t adapt to potential changes, you create high risk situation for yourself
  • introducing several devs require process, BUT will benefit in long run

Always remember, that unsustainable practices will introduce deferred cost

Time and Change

As said, with scale comes need to adopt for more and more fundamental changes, and, if it is not done before-hand, it WILL back-fire hard, because:

  • there are many hidden things, inside program, that will slow-down upgrade
  • it is not enough experience overall in this field
  • it is long and costly process, where you need to JUST do migration alone
  • ---
  • to avoid, generally, invest in incremental updates
  • focus on possibility to overcome any update forward, not just first one, it is constant battle
  • remember that any dependency CAN and most often WILL change(usually because new feature, optimization or bug is found)
  • ideally, you need to reduce that cost of change, by paying upfront and investing in it overtime, OTHERWISE you have risk of been stuck in impossible to change state, where you defer to get more resource, but deferring causes more need of them
    • such problems often slowly become worse overtime, until hitting critical point
  • never do changes for the sake of change
Hyrum’s Law

This law states, that no matter what contract defines, giving long enough time period, all observable behavior of public API will be depended-on by user

  • ex: if you return internal details in error response, that aren’t defined by contract AND stopped, this will break someone, who decided to parse them
  • note: we can mitigate it, but never overcome
  • when deciding to change something, you must include the cost of mitigating breakage problems
    • it works both ways, ex: when using hash-map you can depend on order of items, BUT if you will it will make your code less sustainable; when changing algorithm behind your hash-map library, you eventually will cause bugs in someones code
    • generally, avoid been “clever”, don’t depend on implementation, depend on interface
Policies

Everything need to scale(codebase, dependencies, machines etc), including policies

Scaling problem can be spotted, if process can’t be automated, thus requiring more resources, with more scale

  • remember, that scale of smth(ex: codebase) often will produce non-linear resource requirements(ex: number of maintainers), SO automation is even more important

Examples of bad policies and alternatives:

  • force dependent teams to do migration — perform migration by dedicated team OR team of devs, that introduced need for migration
  • develop in large branches — trunk based development

Examples of good policies:

  • if breaking change wasn’t detected by test suits before merge, this is not a problem of author, this is problem of test owner
  • knowledge sharing process
  • work on improvements of found problems, not just assume that they won’t repeat
    • this will make upstream changes easier and cheaper
  • in general:
    • automate
    • isolate low-level details
    • share expertise
    • introduce improvements constantly
    • keep major policies well known
Shifting left

It is cheaper to fix early problems, so it is important to invest in tools and polices, that will catch any problems on early stage of development

Trade-offs

Have a reason(often backed by data) to do something, don’t don for sake of doing OR “just because”

  • in general, do things because: “you must” OR “it is what’s best, because of…”

When speaking of trade-offs, cost evaluation is key, BUT cost isn’t about money, it is about resources in general(including profits, that won’t be made, because change isn’t done)

  • also remember about societal cost, which is crucial at scale
  • remember, that engineers’ happiness is key metric as well
  • be data driven, BUT always evaluate data and all underline pros/cons, DON’T just trust it

When taking a decision, consider such process, as an example: if I do change X, that will take 2 weeks of my time, gain Y RAM benefit and K CPU loss, will it be beneficial AND, if so, is it the best thing I can spend my time for

  • it is important for company, to share some cost numbers, so it is easy to do calculations

Some decisions can’t be based on measurements, because underlying things are unmeasurable, thus experience AND knowledge + some practices can help to do estimations

Remember, that some decisions may backfire unexpectedly, ex: google built a distributed build system to speed-up build-times, SO engineers bloated dependency lists of their apps(because it is not too much slower anymore to add new stuff in)

  • policies and constraints was added later, to mitigate this problem

To fork or not to fork

  • benefits: control over dependency, possibility to fine-tune to specific needs
  • problem: you now a maintainer, so scalability and sustainability suffers
  • general solutions:
    • if it is short project - fork to move faster
    • if it is timeless project, be careful with forks AND, if forking, keep interface same/similar
Making mistakes

It is ok to make wrong assumptions OR decision, it is bad not to admit them

  • even if decision is data-driven, data can change, so change in direction is needed too

Culture

How to work on teams

Software engineering is all about team work and collaboration, so it is important to build related processes well, BUT you need to align your behavior first and then try to “fix” others

People problems:

  • insecurity - people don’t like to share work in progress, because it might reveal smth bad about them
    • it can also lead to stolen ideas
    • hiding WiP code is harmful, because:
      • you can’t be sure, that you are on right track
        • Many eyes make sure your project stays relevant and on track

      • knowledge sharing is non-existent
      • it is impossible to get external help
      • it blocks “shift left” / early detection of problems
      • your project will die, if you stop working on it, because there is no other contributors
        • that’s why you need co-owners AND docs
  • genius problem - people want to be seen as geniuses/idols in there field
    • you must remember, that always behind big names there is a group of talented people, that was organized by BigName, but we can’t credit only leader of a group
    • this can often lead to stealing credit for group work OR just been a jerk, which is bad and only sabotages development
  • socialization - if people placed in large groups, no one will speak, if people isolated, no on will speak
  • impatience - be patent with each others ;)

main pillars of human interaction and collaborative work:

  • humility - you aren’t center of universe or perfect in any sense, always improve
  • respect - treat people and their achievements with genuine care and appreciation
  • trust - trust that people competent AND can do work by themselfs
  • ---
  • this pillars are important, as prevention of complex situations in human relationships. It is a way to make messy humans to interact easier
  • why? such “small” things will make playing social game easier, thus creating strong relationships

how to achieve pillars:

  • loose your ego, BUT leave self-respect
  • build team ego, by sharing accomplishments
  • don’t try to be main character in the room, even if you owner of some secret knowledge
  • learn how give and take criticism:
    • NEVER criticize personal stuff
    • be constructive
    • make criticism lead to changes
    • be polite
    • ---
    • don’t take criticism personally
    • remember that you != your code

failuers:

  • innovation and risk MIGHT and often WILL result in failure, AND it is ok
    • but, remember to fail as early as possible
  • good way to make failure result in something is doing a postmortem(analysis of situation), BUT it must be good:
    • don’t be pointless
    • don’t be juts to excuse
    • include explanation of what went wrong AND what is going to change
      • this must be followed along
    • common structure: summary, timeline, root cause, impact, action items and their owners to execute, actions items to prevent future problems, learned lesson

notes:

  • it is ok to change your mind, BUT don’t do it to often
    • the best way it to listen, before acting, thus you won’t discover and change too often in other’s eyes
  • let others influence you AND you will be able to influence them
  • choose your battles wisely
  • don’t be afraid to say “I don’t know” OR “I made a mistake”
    • you collab with your team, NOT compete
  • find a culture and work style fit
    • ex: it you can’t work remotely efficiently, just don’t

be googley:

  • be able to solve problems in hard and non-deterministic environments
  • value feedback
  • change status quo
  • put the user first
  • care about team
  • do the right things, even if it is hard

Knowledge sharing

Collectively, organization must always know domain better, than external people, BUT to achieve this org must have experts to have initial knowledge AND mechanisms to share it, such as:

  • culture of asking & answering questions
  • writing docs
  • knowledge sharing events
  • culture of learning
  • culture of admitting that you don’t know smth

challenges:

  • fear of been seen as stupid
  • knowledge fragmentation between teams, that leads to:
    • incomplete general picture
    • information duplication
    • several ways of doing same thing
  • culture of “let me do this for you, because it is faster”
    • results in single point of failure, where single person holds all the knowledge
  • parroting - people mimic behavior of other, without knowing the reason
    • ex: using patterns, just because
  • graveyards - people afraid to touch some part of code OR practice, because it can lead to problems

philosophy:

  • all people start as beginners
  • ”I will do it for you” doesn’t scale
  • single point of failure is always a problem
  • documentation scales well, BUT it must be maintained and can be not sufficient for edge-cases of learning
    • docs help avoid tribal knowledge
    • knowledge holders will still be needed, because they can point to right docs OR right person, actualize docs, help with edge-cases, BUT they can’t be main point of information
  • learn continuously

how to overcome challenges:

  • psychological safety
    • embrace process of trial and error
    • assign mentor per each newcomer(ideally mentor can’t be a direct manager, tech lead or peer)
      • mentor must have enough knowledge
      • mentor is first person to ask, if you don’t know smth
    • facilitate asking questions AND will to answer them
      • this is even more important in group context, so person won’t be afraid to ask question AND someone will answer, without been attacked by other peers
      • to do so:
        • don’t be surprised about question
        • don’t be biased towards person OR try to assault
        • guide in proper direction
        • be patient
        • try help and not showing off
        • establish discussion
        • don’t be pedantic
        • don’t interrupt ongoing discussion
  • always learn
    • ask questions
      • it is ok to struggle a bit, BUT don’t be stuck for too long
    • remember that you can’t know everything
      • seniority != know everything
    • find environment to learn smth new
    • figure out the context behind decisions/design/code etc, first ask why and dive deeper AND only then act
      • you can’t act on thing, if you don’t know the reason why/if they are needed
      • it doesn’t mean that original thing is right
      • document your reasoning
Community knowledge sharing

when you find smth new(in general OR from 1-1 session) write it down and share to community

  • this helps preserving knowledge

there are several complementary strategies, when it comes to knowledge sharing in community:

  • group chat
    • types: topic-based(have large audience), team-based(more specific topics can be discussed, safer for new people)
    • benefits: you can get fast response for some question + other members can find new knowledge from there
    • problems: have no structure and such knowledge hard to preserve
  • emailing list
    • similar to group chat by benefits, BUT scalability problems can be mitigated, by archiving threads AND indexing them via wiki search engine
    • problems: slower then chats, can be noisy
    • notes:
      • remember to refer final answer(if it is several messages, write a wrap-up)
        • this prevents situations, where you don’t know if thread is closed OR just frozen
      • on personal basis, set-up filters and other automations to deal with email flow, BUT as company, restrict emails to send them to target group, not just “CC @everyone”
  • Q&A platform
    • similar to emailing list, BUT with additional UI/UX benefits, such as: custom interface, “mark as answered” etc
    • require additional set-up and maintenance

there are several ways you can teach someone:

  • office hours - events, where engineers gather and discuss some question/problems
    • problems: problematic to organize, sometimes people don’t know what to ask OR some questions hard to discuss in such way(specific, urgent etc)
  • talks and classes - events, where engineers teach others somethings new
    • can be even non work related, just for building the community
    • types:
      • talks - share some knowledge
      • classes - share some knowledge with active participation in class from audience
    • when to create talk/class:
      • subject has a lot of demand
      • subject is stale enough
      • subject involves teacher AND is hard to self-taught
  • documentation - written knowledge, that has main goal to teach reader smth
    • reasons to contribute:
      • update - you can learn somethings new + learn how to contribute to docs
        • it is same rule, as for refactoring, leave place cleaner, that it was before you
      • create - if you created some new flow, discovered new things OR just gain valuable knowledge - share it with others
        • locate properly docs, so info can be found easily
        • ideally, place docs near relevant code OR place links to this docs
    • good docs must have such systems:
      • comments(can be with auto creation of Jira tasks to fix them)
      • review and diffing
      • possibility to mark status of doc(draft, relevant, obsolete etc)
      • be located OR linkable to code
    • how to promote docs (it is time consuming process, that will benefit many people in long run, so it is important, BUT may be hard to promote)
      • save time, by pointing to docs, instead of answering
      • enforce sharing and contribution docs
        • this leads to docs to be noticed, thus leading to moral boost of an author and acts as great incentive
  • code - code is written knowledge too, thus making code clear and readable makes sharing knowledge easier
    • docs and comments are boosters of knowledge sharing via code
    • code reviews are boosters of code quality AND help share new practices between author and reviewer

cultivate knowledge sharing as culture via:

  • respect others
  • create incentives top->bottom:
    • how: make it part of senior position responsibility, make it part of KPIs and review
    • responsibilities: become mentor, share knowledge(code reviews, docs, talks, classes)
  • create incentives bottom->top, so person receives acknowledgement from colleges
    • peer bonuses, rating, public “kudos”

establish canonical source for important info

  • prevents information fragmentation, BUT it is harder to maintain
  • such docs MUST be owned and provide an option for feedback, that will be changed by owners directly
  • such docs can be team, office, org etc based too
  • subtype is dev-guide docs, that can provide insights on how-to develop, what style or tooling is used etc

tips:

  • establish short, stable urls for docs
    • ex from google is “go/”, redirect to URL service, that easy to remember+link and will be always stable, even if underlying url has changed, like go/python
  • if possible, consider creating learning platform
  • automate some knowledge, via shared linter/prettier configs
  • big to small updates must be shared, so author gets acknowledgement AND people can find new info, this can be done via:
    • newsletter
      • note: never spam with letters, keep only relevant info in them
    • communities
      • creating communities prevents info fragmentation
      • note: each community can have different level of formality OR even be non work related
Code Reviews

google’s practices:

  • be picky with first couple reviews of new hires, to establish best practices of how code is written in company, thus preserving it’s consistency
  • for important MRs, require check from certificate person, that has deep domain and language knowledge to make a review
    • such reviewers act as teacher, when reviewing code
    • discussion is welcomed
    • such process is great for docs sharing and explaining
    • be aware that such process is hard to scale, BUT it is worth tradeoff, to achieve knowledge sharing AND consistency
      • part of this process can and should be automated with tooling

Software that enables greater good

As said before, one of key values, that engineer can have is to respect users of it’s product fully and equally

  • remember to think as your user would

all people have some bias by default, often an unconscious one

  • if you don’t have enough representment of each group in your company, it will be biased too
    • Google’s ex: image recognition algorithm, at some point, didn’t recognized black people as people, because of poor training data
  • still, bias is expected, when targeting specific auditory
    • BUT, if you have enough market share, underrepresented groups are great targets
  • ---
  • such problem can lead to loosing reputation & market share

how to prevent bias:

  • trainings
  • proper distribution of stuff AND equal opportunity
    • this will also bring a flow of people to your company
  • always remember your target audience
  • recognize and work toward fixing biases
  • use proper data-sets
  • always question current and future decisions
  • listen for feedback AND be data-driven in your approaches

remember about:

  • phycological safety of each one in the team

Leading a Team

Group of individuals is dysfunctional without a leader, BUT leader can be not one For teach team it is beneficial to have two leaders, with same goals, BUT different perspectives and skill: Manager(PM), Teach Lead

  • team will be dysfunctional, because it will has no aim to move to
  • this is important chapter of individual contributors too, for better understanding of your management
  • both roles can be taken by same person, for smaller teams

PM - role that responsible for aligning company goals with engineering happiness, do plannings etc

  • individual and company goals might not align, so it is hard job
  • ideally, product must have engineering background

Teach Lead - highly technical role, that manages any large tech decision, priorities, tech level of teammates, in scope of the team

  • lead can be IC too, BUT it is often more beneficial to delegate
    • especially on scale

TL+PM - acceptable only for small teams, otherwise person won’t perform good enough

Remember, that it is possible to influence large amount of people, BUT don’t be authoritarian, by:

  • speaking
  • explaining and aligning goals
  • be human, respectful and trustful to everyone
  • not micro-managing
How to be a manager/lead

You don’t have become manager, BUT it may happen so, thus it is helpful to know how to transition

problems with management:

  • it is hard to quantify what manager does
    • to fix, remember that team efficiency AND happiness is what you do
  • bad managers make look all managers bad
    • learn how-to, try and decide wether you like been a manager AND then fully promote

why to become:

  • you limited in amount of code, you can write, BUT you can stere many people under you

problems of managers:

  • micromanagement
  • ignoring low performing teammates
  • ---
  • remember, NEVER over-manage people, try serving your people
    • otherwise you will loose hard to replace workers, that can’t work in such atmosphere
  • don’t forget about social aspect of a team

Traditional managers worry about how to get things done, whereas great managers worry about what things get done (and trust their team to figure out how to do it).

patterns of bad managers:

  • hire people that will be less smart or less ambitious then you, so you will be more important and more irreplaceable as their leader
    • problems: now team can’t operate without you AND overall results are bad
    • overall, try to hire people smarter then you, they can replace you AND will challenge, BUT now you AND your team can grow
  • ignoring long performers, because it is hard for you to fire them
    • problem: team will loose in performance, motivation etc
    • to deal: set-up timeframe, set-up measurable goals, set-up status check-ups
      • it is ok to micromanage, BUT with respect
  • ignoring human aspect of the managing people
  • try to be friend with everyone
    • problem: your subordinates will fill obligated to be your friend, because you are a boss + it is hard to be harsh with your friends
    • better to keep most relationships on the work side of spectrum PLUS some informalities put into it
  • compromise hiring bar for speed
    • problem: team quality degrades, it is costly to hire bad person
  • treat your team as babies, by micromanaging, don’t trusting, disrespect, not giving opportunities etc
    • problem: people behave, as they are treated

patterns of good manager:

  • loose your personal ego, BUT still have self-respect and self-confidence
    • cultivate team ego
    • to achieve: trust people, guide people not take their hand and walk to the destination along with them
    • respect constructive criticism
    • apologize
  • mediate your emotions
    • you must let people know that you know the problem, BUT you can’t overreact
    • your reaction will be mirrored by your team
    • remember, you been constantly watched by your team
  • don’t try to find solution, help person in finding it
  • be a catalyst, by speeding-up the process, BUT not acting in it
    • as a leader, your goal is to build a consensus (ideally by team-member’s hands, BUT, if needed, you can do it yourself)
    • look for roadblocks, it is often your responsibility to help with them
      • often, knowing the right person is a key, SO build a network inside your company
  • remember to also be a mentor
    • junior will take more time to develop smth, it is ok, it will scale your team in future
    • to be mentor you need to: know the process, be able to explain smth, give just enough needed information
  • set clear goals
    • set goals -> let people execute -> check if everything moves in right direction
  • be honest
    • tell that you don’t know smth OR can’t tell
    • give proper feedback, even if it is bad
      • don’t use techniques like compliment sandwich, it is better to say things as is, BUT include emphasis
        • without emphasis person will go into defense mode
  • make your team happy
    • first and most important, ASK people what they need
    • explicitly care about people, their careers etc
  • lower pressure, when persons have personal problems
    • BUT, it is ok to ask them for “compensation” later

tips & tricks

  • delegate, BUT dirty your hands too
  • seek replacement
    • to do: hire people smarter then you, create opportunities to tru “manager shoos”
    • why: you can grow this way
    • DON’T force ICs to become managers, just because
  • act as quick as possible in some situations, like: low performers, bad hiring process etc
  • shield you team from chaos, BUT keep them informed, when needed
  • provide positive feedback too
  • if something can be easily undone - say “yes” to it
  • remember that everyone is different AND needs different approach
    • it mostly comes to: motivation, direction
    • giving direction is about finding what needs to be done AND breaking it into smaller parts, that can be assigned to each teammate
    • givning motivation
      • focus on intrinsic motivation, by providing:
        • autonomy
        • mastery - provide possibilities to improve
        • purpose - help seen the purpose in person’s work

Leading at Scale

When you start leading one team it is natural progression to lead several teams, so this chapter talks about it

  • all best practices from before apply ;)
  • larger scale == higher scope == less engineering knowledge used

three always of leading:

  • always be deciding - you will take more and more decisions, that must be evaluated by tradeoffs
    • focus on finding the pass, that engineers can follow to achieve something
    • to decide:
      • identify the blinders - if progress on decision was made before, gain knowledge about the problem AND identify what blind spots was left by people
      • find and explain, to everyone involved, tradeoffs
      • make a decision AND iterate upon it(identify the blinders, find tradeoffs, make a decision again)
        • always iterate, don’t seek perfect solution
        • notes:
          • some tradeoffs can be represented as triangle, where you can only pick two
          • search for all tradeoffs, even if some thing seems like it is not a tradeoff at all
  • always be leaving - build processes to work autonomously without you, so you could leave and work on a new problem
    • all teams must be self-driving, basically by having good leadership, healthy engineering processes AND positive culture
    • to achieve this:
      • divide the problem space - build loose organization structure, so you could divide problem to subproblems, BUT keep things flexible, so each team could iterate and communicate with each structure
        • each team must have sense of progress, ownership and accomplishments
        • make team own problem, not solution
      • delegate subproblems to leader, let them fail and iterate
        • always ask if you really the only one who can fix it
        • also ask yourself what I can do that others can AND fill such areas with other leaders
          • it is not always possible for some tasks tho
      • iterate by adjusting processes and teams
        • alternatively you can grow further in career ladder
        • don’t micromanage, focus on small, but critical adjustments
          • always listen to people
      • anchor team’s identity by assigning problems to them, not products(product is solution, that can be changed overtime, making team loose identity AND defeat the solution, instead of iterating upon better one for the problem)
  • always be scaling - scaling the team is important, but everything discussed prior results in scaling, SO here will be talked about scaling yourself, meaning proper management of personal time and resources
    • it is common thing, when finishing the project, to gain additional project as result
      • one success results in discoveries in adjacent areas AND more scope to handle with same amount of people
      • basically you need to handle more to scale, BUT hiring can’t keep-up, so you must know how to compress prior problem
    • main problem is that, with scope growth you become more and more reactive on stuff that is “thrown at you”, SO differentiate urgent from important by:
      • delegating
      • scheduling dedicated time to respond OR to do important
        • based on your role, more time to do for IC and more time to respond for manager
      • track tasks through some system
      • just drop incoming request
        • you will do it anyway, because there are too many of them, so do it intentionally
        • one strategy is to breaking things into three groups: 20% of non-important, 60% of somewhat-important, 20% of important; drop 80% and focus only on what is important
          • 60% will be either picked-up OR go back to top 20%
    • focus on protecting your energy
      • partially you will adapt to work speed over time, BUT it is also important to be proactive here, so:
        • take real vacations - take week+ vacations AND avoid most of contacts with work
        • don’t mix personal and work -don’t take laptop to home, disable notifications on phone etc
        • take breaking when working
        • take sick leaves AND mental health day-offs when needed

Measuring Engineering Productivity

To be fully data-driven focus on engineers productivity too

why?

  • scaling team is a way to scale company, BUT growth is not linear, because of overhead, SO by scaling individual engineers you scale company in linear-ish fashion

note: it must be done naturally AND at adequate cost

process:

  • identify what need to be measured
    • it takes time, resources, makes team slower OR behave unnatural, when measuring, so you need to measure only what is needed
    • always ask:
      • what result is expected? - to prevent biased results
      • what actions will be taking if result is positive/negative? - to check if it makes sense to measure something, that will not result in any change
        • change can also mean maintaining status quo
        • sometimes you can identify what really will result in change AND measure that thing
      • who will take an action and when?
        • responsible person must be assigned
        • action must be based on relevant data, so time is important
      • do you trust this [source of data]?
        • seek for qualitative or/and quantitative analysis
    • it is ok not to measure when change:
      • is to costly
      • is irrelevant, because of other incoming changes
      • is going to be made anyway
      • can’t be measured precisely

what metrics to choose:

  • one possible framework is Goals->Signals->Metrics, where
    • Goal - what you wan’t to know, not how
      • this must be a full list of goals, that covers all trade-offs
      • common areas to consider:
        • code quality
        • attention of engineer(flow state, notification distraction, context switch)
        • intellectual complexity(base complexity, unknown unknowns)
        • tempo and velocity
        • satisfaction
    • Signal - how would you know that result is achieved
      • signal can be thing that we want to measure, but it is impossible
      • it is needed to establish signals first to prevent using only available measurements, think what ideally you would need
    • Metric - basically a measurable alternative to Signal
      • something close enough, that can be viewed as signal
      • by defining what to measure upfront AND that this thing is suitable for our needs, we can prevent doing additional metrics after the result and the result will be more trustable
      • metrics can be combined to find deeper details about signal
    • ---
    • framework allows us to understand what we measure AND why, by providing traceability

data

  • you can collect data in two ways:
    • quantitative - collect as much as possible, BUT with small insight
    • qualitative - collect rather small sample, BUT with reach insights

action

  • after measurements are done TAKE ACTIONS, try to keep them tool-driven, so it is good tools that force processes, NOT just recommendations

notes:

  • centralized “data team” can be much more effective that forcing each team to do their own data research
  • if qualitative metric don’t match quantitative -> quantitative is incorrect in some way

Processes

StyleGuides and Rules

Define set of strict rules, that aren’t negotiable AND guidances, that are recommendations

StyleGuides:

  • treat your style guide as combination of style(prettier) AND conventions(linter)
    • conventions must be strict and understandable
  • main goal is clear, sustainable code
  • it is important to lean towards reuse of industry standard rules, BUT creating something in-house, as addition is important sometimes

Why?

  • to encourage good and discouraged bad behavior
    • good and bad is aligned with company goals and can differ
  • to establish common language AND patterns

Creating the Rule:

  • ask what goal you need to achieve AND why
  • google approach is to create strict rules, that will be less flexible, BUT more oriented towards time sustainability and disallow arguments between engineers on how to do things, so here are common themes in their rules
    • pull their weight - add only important rules, that can be less known to avoid mental overhead AND unnecessary large amount of rules
    • read-first - code is written once, read many times
      • proper names, easy to follow flow of code, easy to understand intent and what is happening, clear API contracts and abstractions
    • consistent - tooling, conventions etc must be kept consistent for ease of adoption
      • engineers spend less time on thinking about how to do things
      • code is universally readable
      • scale: easier to refactor, easier to provide tooling, less time-to-learn when changing teams or working in different codebases
      • ideally adopt all old code to new standards, BUT ideal consistency costs too much
        • use code-generation tooling to do migrations
        • one solid choice is to enforce consistency on each level first and then on level higher: file, team, project, monorepo, company
      • be, as much as possible, consistent with external community of devs
        • it is great longterm thinking
    • avoid error-prone and hard to understand constructs - prevent hard to understand things from appearing in codebase
      • if you understand it !== other people understand them now OR still will in future
      • easy code easier to debug
    • be practical
      • if performance is critical, be inconsistent
      • if you use external lib, let it be inconsistent, it is not owned by you
      • etc

What style guide should contain:

  • rules to avoid danger
    • outline what functionality of language should or shouldn’t be used from technical perspective
    • prevent common errors and problems
    • provide pros+cons and rationale for rule
  • rules to enforce best practices
    • provide best practices to keep codebase healthy, maintainable and consistent
    • improve readability of code
    • prevent new language features and libraries from getting adopted too soon
      • first study it on small scale AND only after go for broader adoption
  • rules to force consistency
    • this are some rules, that have small to no effect, BUT was chosen to prevent any debate between co-workers
      • ex: tab vs space, 2 vs 4 indentation etc
  • ---
  • don’t include in your style guide, BUT remember:
    • don’t be clever
    • don’t reinvent the wheel
    • don’t fork codebase

Changing the rules:

  • it is important to keep rules up-to-date with industry, language, practical and automated, SO when it becomes a burden to do so, rule can be changed
    • it can’t happen to often, because rate of migrations won’t be realistic at all, SO evaluate each new rule and include reasoning with them for future self(you would be able not only understand why to keep, BUT also see if rationale no longer valid)
  • reason for change must be real, not potential
  • change must be: necessary or beneficial and relatively cheap to implement + it MUST be agreed on by community first, but finally approved by style guide owner
    • make decision based on trade-offs, NOT preferences
  • note that rules CAN and sometimes MUST have exceptions

Guidance (aka the should’s)

  • basically guidance is form of passing collective knowledge of some topic, that doesn’t have to be strictly followed, BUT it is often beneficial to do so
  • some guidance can be short, in form of a tip that can be read quickly, well remembered, broadly used AND even shared as daily/weekly advice
  • some guidance can be just a reference/link/example, that can explain some hard topic
    • links for internal lib docs is great way to reduce duplication

Apply rules:

  • types:
    • social:
      • docs - keep them up-to date
      • classes
      • code reviews
      • spreading rules via some info-channel
    • technical: linter, prettier, compiler etc
      • such tools will make process of enforcement scalable and predictable
      • some problems are purely social and vaguely defined, so it is impossible, costly or will result in flacky results
      • allow some rules to be chosen on per-project basis, if they aren’t strict

Code Review

Code review is process of review changes to codebase, but non-author of change, before merging this change to codebase

  • this can be interpreted in different ways, in Google this means that everyone is involved in process AND every change is involved too
  • basically you need process(this chapter) AND tooling(later on)

process:

  • if code is comprehensible and can be well understood by reviewer -> LGTM from any reviewer
  • if code has LGTM + it is appropriate -> approve from code owner(gate keeper of some codebase)
  • if code has LGTM + it compiles with style guide -> readability approve from readability expert
  • ---
  • self-review, more that one LGTM, review from single person who is readability expert and codeowner is used to add flexibility and scalability to process
  • owners - keeping small number of documented, inside some file, codeowners, that act as gate keepers for some particular part of code, is a great way to introduce strict control for quality to each individual part, because it guarantees review from domain expert
    • such files is great way to document
    • by maintaining tree-like structure you could easily manipulate code on different levels, BUT don’t bother too much people for review

benefits - while common benefits are understood, it is important to look deeper AND maintain important role of reviews, keep it as part of culture(not just nice sanity check, that is actually LGTM)

  • code correctness:
    • check for: structure, design, bugs, tests, efficiency etc
    • the earlier you find bug the cheaper it would be
    • comments can’t be opinionated(AT LEAS PRETEND ;))
  • comprehension of code
    • basically code review is first every inspection of code, made by someone, other then author, THUS it is a way to understand if the code is comprehendable
      • it is crucial, because code is read more often, than written
      • from this perspective, reviewer should also be interested in code’s health, meaning been it’s maintainer
    • try to view all questions as valid AND worse considering, because they will happen again over time
      • this means that code might be too complex AND would benefit from some change
  • code consistency
    • code will be read by others, refactored by automated tooling, SO it not only must be well understood, but also standardized
    • it is also used to ensure that code is “simple”
  • psychological and cultural benefit
    • code review makes author think about greater good, disregarding their own style
    • also it is a great way to learn and receive criticism
      • so keep feedback itself radical, BUT in natural box
    • it is a way to mitigate impostor syndrome, by receiving positive feedback
    • it is a way to prevent too much of cutting corners
      • overall, keep your time to reflect on your changes before review
  • knowledge sharing
    • don’t hesitate to sprinkle some “FYI”s into your review process
    • process of knowledge sharing is bi-directional
    • also, using commit+MR+comment combination you can find some clarification on rationale behind the change
  • ---
  • benefits can apply for both sides of process
  • notes:
    • keep process lightweight enough, so it is scalable
    • this benefits are so important, that it is worth to slow time-to-production of change rate AND keep this processes rather strict
    • don’t strive for perfect code, approve if code makes codebase better

best practices (required so process is fast and smooth):

  • be polite and professional
    • ask why something is done in such way, don’t go to conclusion
    • keep discussion professional
    • accept author’s view, if it has valid points
    • don’t view code as yours
    • take your time to explain and argument your point of view ON PER COMMENT BASIS
      • don’t resolve until agreed upon
  • give feedback all at once
  • respond as quick as possible
  • write small changes
    • it is faster to review
    • it is easier to understand
    • review will be more qualitative
    • BUT, remember that:
      • small changes might hide larger picture
      • can be impossible (feature flags, trunk based development and proper processes can fix it)
      • set of small changes can be slower then one large
  • write good description and commit names
    • include all changes into it, BUT keep it small
    • it will help with bug fixing
  • keep number of reviewers low
    • it is just not worse it to have too much people review at once
    • but if you have several reviewers, try to focus on different things
  • automate
    • static analysis, proper CI/CD, tests
    • you can even reject possibility of review, before all pre-commit checks are green

types of review (each type need to be treated differently):

  • new code
    • evaluate if it will stand the time AND scale properly
    • before any large new code additions do design(or similar process, like: ADR, PRD etc)
      • code must be done by design
    • code must be tested
    • it must have defined owners AND proper docs AND proper CI/CD etc configs
  • behavioral changes OR improvements
    • if code deleted - great, if not, check if it is improving the codebase
    • new code practices apply here too
    • tests must be green
    • if change is optimization - provide reasoning AND benchmarks
  • bug fixes OR rollbacks
    • fix only found bug AND change tests to account for that bug
    • any change that can cause a rollback must be easy revertable and atomic
    • rollback can be also done by effected downstream consumer
  • refactoring OR large-scale changes
    • some changes are generated by machine, BUT it still must be reviewed
      • low risk - owner of codebase
      • high risk - owner of codebase + owners of chunks
    • chunk owner can review only their chunks AND comment only on specifics of their code

Documentation

Docs are important process, that need to be built around proper tooling and processes, so it is easy and scalable to write docs from engineers to engineers

  • lacking docs is often main pain-point in day2day work, THAT can’t be fixed by PM, tech lead or teach writer
  • without processes it won’t be realistic to achieve good docs, LIKE it was with testing prior

documentation is beneficial cause:

  • code is understandable
  • onboarding is smoother
  • objectives are clear
  • it helps with establishing proper API/interface
  • it helps with non-trivial places
  • it makes public API looks more professional
  • you can refer to the docs, instead of re-answering questions
  • ---
  • benefits aren’t upfront, but will help future your and other maintainers in the future

it is hard to write docs, BUT we don’t need to avoid it, we need to invest in tooling

  • make it easy to write docs, BUT docs must be easy to read

documentation must:

  • have rules, standards, like code itself
    • in general keep it close to the code(by rules, by location, by content, source control, reviews etc)
  • be owned AND maintained
  • have single source of truth AND avoid duplicates
  • be easily searchable
  • evaluate docs from time to time for been actual
  • be fixable(or at least take tickets to be fixed) by reader
  • (ideally) docs must be changed alongside the code

understand your audience:

  • if it design document - focus on decision making, if it is guide - focus on explaining topic etc
  • treat docs as test, not your main activity, BUT very important
    • remember, you don’t need to be great writer, just try your best to express your knowledge
  • keep level of explaining acceptable to your audience, BUT, if possible, appeal to broader one
  • take your time to refine document(remove duplicates, condense etc)
  • if possible: provide TLDR
  • keep docs consistent
  • keep public docs of public API separate from implementation docs