Welcome to the open-source repository of the North app!π
The North app automatically calculates your carbon footprint by connecting to other services and apps in your life and translating activities from these apps and activities to greenhouse gas emissions.
The app is private-by-design: data will stay on device, unless the user explicitly gives consent. This code is maintained by Tomorrow.
Feel free to watch the presentation our Founder Olivier gave to the CopenhagenJS meetup, explaining what a JavaScript developer can do to combat climate change. If you have any questions, want early access to the app or just want to hang out with people fighting climate change with code, join our Slack community.
The company behind the North app builds tech to empower organisations and individuals to understand and reduce their carbon footprint.
We're hiring great people to join our team in Copenhagen. Head over to our jobs page if you want to help out!
./co2eq
: carbon models./integrations
: contains all integrations./integrations/img
: contains all integration logos./playground
: source code of the playground./definitions.js
: constant definitions
You can help by:
- Helping us find, add and improve integrations with 3rd party services
- Helping us find, add and improve our Life Cycle Assesment / Carbon footprint data
- Giving us ideas, feedback and reporting bugs
We wrote a little article about what we believe makes a great integration. However, that shouldn't stop you from doing an integration which is interesting and useful to you! Our only requirement is that it can be quantified in greenhouse gas emissions.
Integrations can rely on an API or even on scrapers if necessary.
Here is the list of current 3rd party integrations: Official integrations:
βοΈ Tripit (tracks most airlines!)- β‘ Barry
- π Tesla Cockpit
Community-supported integrations:
βοΈ Ryanair (contributor:lauvrenn)βοΈ Wizzair (contributor:lauvrenn)- π Rejsekort
- π Trainline (contributor:liamgarrison)
- π Transport for London (contributor:liamgarrison)
- β‘ Sense (contributor:snarfed)
- β‘ Linky (contributor:bokub)
- β‘ Γrsted (contributor:felixdq)
- π§ Outlook (contributor:baywet)
- π Renault ZoΓ©
- π Uber (contributor:willtonkin)
- π Automatic (contributor:lauvrenn)
- π MinVolkswagen (contributor:folkev0gn)
You can suggest a new integration here.
If you don't have an idea on your own or prefer to debug an integration, you can find integration suggestions and bugs in the issues.
To make it easy for anyone to help out, a development playground is available:
First, run yarn
to install dependencies at the root of the repository.
Next from the playground
folder, run yarn serve
to start the playground and point your browser to localhost:3000 to get started.
An integration gathers activities from a 3rd party datasource. To this end, 3 async methods need to be exported:
async function connect(requestLogin, requestWebView) {
const { username, password } = await requestLogin();
// ...
return newState;
}
async function collect(state = {}, logger) {
// ...
return { activities, state: newState };
}
async function disconnect() {
// ...
return newState;
}
The connect
method is used to ask for user credentials (OAuth flows are also supported).
The collect
method is called periodically (typically every few hours) to fetch new activities.
As the methods are pure, and to avoid re-asking the user for credentials everytime the collect
method is called, a state
object can be used to persist information (such as password, tokens..) across collect
s.
Activities require a certain formatting:
{
id, // a string that uniquely represents this activity
datetime, // a javascript Date object that represents the start of the activity
endDatetime, // a javascript Date object that represents the end of the activity. If the activity has no duration, set to "null"
distanceKilometers, // a floating point that represents the amount of kilometers traveled
activityType: ACTIVITY_TYPE_TRANSPORTATION,
transportationMode, // a variable (from definitions.js) that represents the transportation mode
carrier, // (optional) a string that represents the transportation company
departureAirportCode, // (for plane travel) a string that represents the departure airport, IATA code
destinationAirportCode, // (for plane travel) a string that represents the final destination airport, IATA code
departureStation, // (for other travel types) a string that represents the original starting point
destinationStation, // (for other travel types) a string that represents the final destination
participants, // (optional) the number of passengers (for car and motorbike travels)
}
{
id, // a string that uniquely represents this activity
datetime, // a javascript Date object that represents the start of the activity
endDatetime, // a javascript Date object that represents the end of the activity. If the activity has no duration, set to "null"
activityType: ACTIVITY_TYPE_LODGING,
hotelClass, // a variable (from definitions.js) that represents the class of the hotel
countryCodeISO2, // a string with the ISO2 country code that represents the country of the hotel
participants, // (optional) the number of people sharing one hotel room
hotelName, // (optional) a string that represents the name of the hotel
locationLon, // (optional) the longitude of the location of the hotel
locationLat, // (optional) the latitude of the location of the hotel
}
{
id, // a string that uniquely represents this activity
datetime, // a javascript Date object that represents the start of the activity
endDatetime, // a javascript Date object that represents the end of the activity. If the activity has no duration, set to "null"
activityType: ACTIVITY_TYPE_ELECTRICITY,
energyWattHours, // a float that represents the total energy used
hourlyEnergyWattHours, // (optional) an array of 24 floats that represent the hourly metering values
locationLon, // (optional) the longitude of the location of the electricity usage
locationLat, // (optional) the latitude of the location of the electricity usage
}
{
id, // a string that uniquely represents this activity
datetime, // a javascript Date object that represents the start of the activity
label, // a string that represents the transaction
activityType: ACTIVITY_TYPE_PURCHASE,
merchantDisplayName, // (optional) a string that represents the merchant where the purchase was made
lineItems, // an array with specific items that can be either in monetary or amount-based form: [{ identifier: XX, value: 2.1, unit: XX}] where `identifier` is a key from footprints.yml and `unit` a valid unit from definitions.js. See purchase/index.test.js for examples.
bankDisplayName, // (required for integrations with banks) a string that represents the banks' name
bankIdentifier, // (required for integrations with banks) a string that uniquely represents the bank.
}
{
id, // a string that uniquely represents this activity
datetime, // a javascript Date object that represents the start of the activity
label, // a string that represents the meal
activityType: ACTIVITY_TYPE_MEAL,
lineItems, // (required if the activity contains ingredients) an array with an object [{ identifier: xx, value: 2.1, unit: 'kg'}] where `identifier` is a key from footprints.yml and `unit` a valid unit from definitions.js
mealType, // (required if the activity is a meal type) a string with the value being one of the meal type options in definitions.js
}
Our current models and Life Cycle assessments are accessible here. If you know better sources, please contribute with your knowledge.
If you want to add individual items or ingredients, this is done here. Ideally, the studies used should be as global as possible and it's even better if they're systemic reviews (multiple studies in one!).
Currently, CO2 models must expose the following variables:
export const modelName = 'model name'; // Specify name of the model
export const modelVersion = '0'; // Specify the current model version
export const explanation = {
text: 'description of the model',
links: [
{ label: 'Source Name (year)', href: 'link to source' }
],
}; // Description and sources of the model
export const modelCanRunVersion = 0; // Specify the current version of the can run function
export function modelCanRun(activity) {
const {
...
} = activity; // Deconstruction of activity for relevant fields
if (fields are present) {
return true;
}
return false;
} // Verifies that an activity trigger the model to compute CO2 footprint
export function carbonEmissions(activity) {
// ...
return co2eqEmission
}
} // Computes the CO2 footprint of the activity
After each update of a CO2 model, its version, controlled by variable
export const modelVersion = '0';
must be incremented.
Because a number of apps/sites do not provide an API to access data but send emails instead (eg: e-commerce), you can also implement email parsers that will be run on each email imported by email integrations.
To do so add a new parser in integrations\digital\parsers
and implement the following method:
export function evaluateEmail(subject, from, bodyAsHtml, sendDate) {
// whatever code needed for the detection
return { // return the activity parsing the email discovered
id: `IKEA-${orderMatches.length > 1 && orderMatches[1]}`,
datetime: sendDate,
label: `IKEA order ${orderMatches.length > 1 && orderMatches[1]}`,
activityType: ACTIVITY_TYPE_PURCHASE,
merchantDisplayName: 'IKEA',
lineItems: [
{
id: PURCHASE_CATEGORY_STORE_FURNISHING,
value: parseFloat(priceMatches[1]),
countryCodeISO2: currencyAndCode.code,
unit: currencyAndCode.cur,
},
],
};
}
Note: if you are implementing an email integration don't forget to run
getActivitiesFromEmail
fromintegration/digital/parsers/index
for each email discovered and return the activities found by the parser in addition to your email activities. Note: if you are implementing an email integration you don't add any email activites for each email (digital footprint), only activities generated by the parsers will be added from the content found in emails. This is because it'd impact the performance of the application and it's been decided to disbale adding email activities at the moment. More information, the email emission factor is available here.
Please add an issue here or directly in the app.