Practical Clean Code

BarcelonaJS - 2018-10-22

Israel Saeta Pérez (@dukebody)

TravelPerk

mission:

manage complexity

projects delay/failure

sneaky bugs

developer suffering & quit

contagious mess

What is clean code

  • reads like prose
  • makes it hard for bugs to hide
  • can be enhanced by a developer other than its original author
  • provides one obvious way to make things

1. the prose-programming process

  • Step away from your keyboard
  • Write the logic you want to implement in
    ​a paper
  • Write for non-technical users
  • Use multiple different levels of detail until you get to something you believe you can implement

 

  • Implement each different piece of logic as a separate function

can a person buy beer?

a person can buy beer if it is at least 18 years old

 

  


to compute the age of a person from its birthdate

take the difference from current year and its birthdate year

remove one if its birthday has not yet passed this year

the birthday of a person has passed if the month of its birthday is larger or equal to current month, and the day of its birthday is larger or equal that the current day

can a person buy beer?

function canBuyBeer(person) {
    return person.age >= 18;
}


class Person:
    get age() {
      const birthdate = this.birthdate;
      const now = Date();
      const year_diff = now.getFullYear() - birthdate.getFullYear();
    
      if (this.hasAnniversaryPassed(now, birthdate)) {
        return year_diff;
      } else {
        return year_diff - 1;
      }
} 

function hasAnniversaryPassed(now, anniversary) {
    return now.getMonth() >= anniversary.getMonth() 
            && now.getDate() > anniversary.getDate();
}

2. use meaningful variable and function names

  • Avoid too short names in general (<= 2 chars)
  • Avoid too generic names like stuff, tmp, list, do, process, etc.
  • Take into account the variable/function scope
  • If it is difficult to name, it might be a bad idea
// bad
let ret = [];
for (const l of legs) {
    const s = l.getFirstSegment();
    ret.push(s);
}
return ret;


// better
let firstSegments = []
for (const leg of legs) {
    const segment = leg.getFirstSegment();
    firstSegments.push(segment);
}
return firstSegments;
// bad, too generic variable names
function getPopularity() {
    let popularity = {}

    allPopularity = db.open("popularity_db.csv")
    for (aLine of allPopularity) {
        let tmp = aLine.split("\t")
        if (tmp.length === 2) {
            popularity[tmp[0]] = int(tmp[1].trim())
        }
    }
    return popularity
}


// better
function getPopularityByAirport() {
    let popularities = {}

    popularityDB = db.open("popularity_db.csv")

    for (line of popularityDB) {
        let values = line.split("\t")

        let correctLine = values.length === 2
        if (correctLine) {
            airportCode = values[0]
            airportPopularity = parseInt(values[1].trim())
            popularities[airportCode] = airportPopularity
        }
    }
    return popularities
}

3. break long nested functions into smaller ones

reduce cognitive complexity

detect duplicate trips

Given a trip, its duplicates are those from the same company that have the same signature

// bad
function getDuplicatedTrips(trip) {
    // calculate signature for trip
    let signature = '';
    for (const leg of trip.legs) {
        for (const segment of leg.segments) {
            signature += getSignature(segment);
        }
    }

    // get trips candidate for being duplicate
    const company = trip.company;
    const candidateTrips = db.trips.filter(company=company);

    // calculate signature for each trip
    let candidateSignatures = [];
    for (const candidateTrip of candidateTrips) {
        let candidateSignature = '';
        for (const leg of candidateTrip.legs) {
            for (const segment of leg.segments) {
                candidateSignature += getSignature(segment);
            }
        }
        candidateSignatures.push(candidateSignature);
    }

    // get the trips for which the signature matches
    let duplicatedTrips = [];
    for (let i = 0; i < candidateSignatures.length; i++) {
        let candidateSignature = candidateSingatures[i];
        if (candidatSsignature === signature) {
            duplicatedTrips.push(candidateTrips[i]);
        }
    }

    return duplicatedTrips;
}

DETECT duplicated trips

// better
function getDuplicatedTrips(trip) {
    const signature = getTripSignature(trip);
    const candidateTrips = getCandidateTrips(trip);
    return candidateTrips.filter(trip => getTripSignature(trip) === signature);
}


function getTripSignature(trip) {
    let signature = '';
    for (const leg of trip.legs) {
        for (const segment of leg.segments) {
            signature += getSignature(segment);
        }
    }
    return signature;
}


function getCandidateTrips(trip) {
    const company = trip.company;
    return db.trips.filter(company=company);
}

DETECT duplicated trips

COMPUTE TOTAL INVOICE AMOUNTS FOR MONTH

Get all invoices for the given month and add up their amounts

// bad
function get_total_invoice_amounts(year, month) {
  // get inovices for given month
  const start = Date(year, month, 1);
  const next_month = month != 12 ? month + 1 : 1;
  const end = Date(year, next_month, 1);
  const invoices = db.invoices.filter({
    created_at__gte: start,
    created_at__lte: end
  });

  // sum amounts for invoices
  let total = 0;
  for (const invoice of invoices) {
    for (const item of invoice.items) {
      total += item.amount;
    }
  }
  return total;
}

Compute total invoice amounts for month

// better
function get_total_invoice_amounts(year, month) {
  invoices = get_invoices_for_month(year, month);
  amounts = invoices.map(invoice => get_amount(invoice));
  return sum(amounts);
}

function get_invoices_for_month(year, month) {
  const start = Date(year, month, 1);
  const next_month = month !== 12 ? month + 1 : 1;
  const end = Date(year, next_month, 1);
  return db.invoices.filter({ created_at__gte: start, created_at__lte: end });
}

function get_amount(invoice) {
  // should be internal to class invoice
  const amounts = invoice.items.map(item => item.amount);
  return sum(amounts);
}

function sum(elements) {
  return elements.reduce((element_1, element_2) => element_1 + element_2, 0);
}

Compute total invoice amounts for month

clarify conditionals

// OK
if (['create', 'update', 'delete'].includes(action)) {
    check_permission(user, 'write')
}


// better
is_write_action = ['create', 'update', 'delete'].includes(action)
if (is_write_action) {
    check_permission(user, 'write')
}


// even better
function is_write_action(action) {
    return ['create', 'update', 'delete'].includes(action)
}

if (is_write_action(action)) {
    check_permission(user, 'write')
}

4. hide details safely

with abstraction and encapsulation

abstraction

Manage complexity by separating the (public) object interfaces from their (hidden) implementation details

encapsulation

Bind together data and operations, allowing only operation through certain methods to prevent misuse

Instead of accessing objects attributes directly, create business methods and use only those, transferring defensiveness from the caller to the callee

// bad
// caller needs to know internals
for (const leg of dog.legs) {
    leg.move(10)
}



// better
dog.move(10)

class Dog {
    move(forward) {
        for (const leg of self.legs) {
            leg.move(forward)
        }
    }
}

ask dog move forward

(hide internal structure)

// bad
function getTotalInvoicesAmount(invoices) {
    let total = 0
    for (const invoice of invoices) {
        // exposes internal structure!
        for (const item in invoice.items) {
            total += item.amount
        }
    }
    return total
}


// better
class Invoice {
    get amount() {
        return sum(this.items.map(item => item.amount))
    }
}

function getTotalInvoicesAmount(invoices) {
    let total = 0
    for (const invoice of invoices) {
        total += invoice.amount
    }
    return total
}

hide internal structure

// bad
function pay(product, wallet) {
    // touches internal data!
    wallet.money -= product.price
}


// better
function pay(product, wallet) {
    wallet.take(product.price)
}


class Wallet {
    take(price) {
        if price > this.money:
            throw Error('Not enough money')
        // ...
    }
}

avoid impossible states

(negative money amount)

// risky
const price = {
    'amount': 13,
    'currency': 'EUR'
}


// better
class Price {
     constructor(amount, currency):
        if (this.amount < 0) {
            throw Error('Amount cannot be smaller than zero')
        }
        this.amount = amount

        if (!ALLOWED_CURRENCIES.includes(currency)) {
            throw Error(`Unknown currency ${currency}`)
        }
        this.currency = currency
    }
}


const price = new Price(13, 'EUR')

avoid impossible states

(negative amount, invlaid currency)

6. start simple and do continous refactoring

  • Avoid premature flexibility or features because You Arent Gonna Need It
  • Keep the design short and simple (KISS)
  • Avoid premature abstraction: duplication is far better than the worng abstraction (wait for three examples)
  • Easy to refactor to add features later if needed
  • Scout rule: Always leave code cleaner than you found it

References

thanks! :)

Practical Clean Code - JS

By Israel Saeta Pérez

Practical Clean Code - JS

  • 2,131