Skip to content

A JavaScript library for interacting with farmOS over API.

License

Notifications You must be signed in to change notification settings

nextechit/farmOS.js

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

farmOS.js

Licence Last commit Twitter Chat npm

An npm package for fetching data from a farmOS server.

⚠️ WARNING: This library is still considered unstable and only intended for use in farmOS-client (Field Kit) ⚠️

Installation

To install using npm (the Node Package Manager):

$ npm install farmos

Usage

To create an instance of farmOS.js:

import farmOS from 'farmos';

const host = 'https://farm.example.com';
const clientId = 'my_clientId'; // Defaults to 'farm', but apps should use their own OAuth Client.
const farm = farmos(host, {clientId});

Authorization

Before farmOS.js can make requests to a farmOS server, the user must first authorize the client with an OAuth2 token. This token will be used to authenticate requests to the farmOS server.

The simplest authorization flow uses a username and password. farmOS.js implements this with farm.authorize(). More advanced use cases may use a different authorization flow to retrieve an OAuth2 Token. These use cases may use the getToken and setToken methods as described in Token State to authorize a farmOS.js client.

For more information on using the farmOS OAuth2 Authorization server see the API documentation.

.authorize()

const username = 'FarmerSteve';
const password = 'XXXXXXXXXXX';
const scope = 'farm_info'; // Defaults to 'user_access'
farm.authorize(user, password, scope)
  .then(token => localStorage.setItem('token', token));

Token State

Uses cases that don't require the user to login at the beginning of each session must manage their own token state outside of the farmOS.js client (also referred to as "offline access".) To make this possible, farmOS.js allows clients to be instantiated with optional getToken and setToken methods.

getToken allows the farmOS.js client to be instantiated with an existing OAuth2 token. This method will be called before the client makes a request to ensure it is using the latest OAuth2 token. This method is also exposed as farm.getToken(), which can be useful if no optional getToken method was provided when the client was instantiated.

The setToken method is called after an access_token expires and is refreshed using the refresh_token. Because an access_token could expire at at anytime during the lifetime of the farmOS.js client, it's important this method saves new OAuth2 tokens to the same state used by the getToken method.

farmOS Field Kit saves state to the browser's local storage:

const getToken = () => JSON.parse(localStorage.getItem("token"));
const setToken = token => localStorage.setItem("token", JSON.stringify(token));
let client = farmOS(host, { clientId: 'farm_client', getToken, setToken });

⚠️ WARNING ABOUT PARALLEL REQUESTS ⚠️

This library takes certain measures to ensure that a batch of requests, if executed in parallel with the same farm_client instance, will refresh the OAuth token only once, and process subsequent requests using the new token. This cannot be guaranteed, however, for parallel requests made with separate instances of farm_client. Such requests may be subject to race conditions, which could cause many of them to fail because they didn't use the newest token. It is the responsibility of your application to manage requests so they either execute parallel requests using the same farm instance, or avoid such parallel requests altogether.

Logs

A log is any type of event that occurs on the farm, from a planting to a harvest to just a general observation.

Here is an example of what one would look like as a JS object:

const log = {
  id: '1',
  name: 'some log name',
  type: 'farm_observation',
  timestamp: '1519423702',
  done: '1',
  notes: {
    value: '<p>some notes</p>\n',
    format: 'farm_format'
  },
}

Methods for getting, sending and deleting logs are namespaced on the farm.log property.

.get()

Use the .get() method to retrieve a single log as a JavaScript object, or an array of objects, which can be filtered:

// Leave the parameter undefined to fetch all available logs
farm.log.get()
  .then(res => console.log(`Log #${res[0].id} is called ${res.list[0].name}`))

// Accepts a number for the id of the log you wish to fetch
farm.log.get(123)
  .then(res => console.log(`Log #123 is called ${res.name}`))

// Accepts an array of numbers for a collection of logs
farm.log.get([123, 456, 789])
  .then(res => console.log(`Log #${res[0].id} is called ${res.list[0].name}`))

// Pass an options object to filter the results
farm.log.get({
  page: 2, // default === null
  type: 'farm_observation', // default === ''
}).then(res => console.log(`Log #${res[0].id} is called ${res.list[0].name}`))

The options object can have two properties: page is the page number in the sequence of paginated results, starting from 0 and in batches of 100 logs; type filters the results by log type.

The four default log types are:

  • farm_activity
  • farm_harvest
  • farm_input
  • farm_observation

Other log types may be provided by add-on modules in farmOS.

.send()

Send can be used to create a new log, or if the id property is included, to update an existing log:

farm.log.send(log)
  .then(res => console.log(`Log was assigned an id of ${res.id}`));

.delete()

THIS METHOD HAS NOT BEEN FULLY DEVELOPED YET AND MAY NOT WORK

// For now, just an example of what it should look like eventually
farm.log.delete(123);

Assets

Assets are any piece of property or durable good that belongs to the farm, such as a piece of equipment, a specific crop, or an animal.

Here is an example of what one would look like as a JavaScript object:

{
  id: '1',
  name: 'Brunhilde',
  type: 'animal',
  animal_type: {
    id: '12',
    resource: 'taxonomy_term'
  },
  description: {
    value: '<p>some notes</p>\n',
    format: 'farm_format'
  },
  archived: '0',
}

Methods for getting, sending and deleting assets are namespaced on the farm.asset property.

.get()

Use the .get() method to retrieve a single asset as a JavaScript object, or an array of asset objects, which can be filtered:

// Leave the parameter undefined to fetch all available assets
farm.asset.get()
  .then(res => console.log(`Asset #${res[0].id} is called ${res.list[0].name}`))

// Accepts a number for the id of the assets you wish to fetch
farm.asset.get(123)
  .then(res => console.log(`Asset #123 is called ${res.name}`))

// Pass an options object to filter the results
farm.asset.get({
  page: 2, // default === null
  type: 'animal', // default === ''
  archived: true, // default === false
}).then(res => console.log(`Asset #${res[0].id} is called ${res.list[0].name}`))

The options object can have two properties: page is the page number in the sequence of paginated results, starting from 0 and in batches of 100 assets; archived is a boolean which determines whether to retrieve assets which the user has chosen to archive; type filters the results by asset type.

Some common asset types include:

  • animal
  • equipment
  • planting

Other asset types may be provided by add-on modules in farmOS.

.send()

Send can be used to create a new asset, or if the id property is included, to update an existing asset:

farm.asset.send(asset)
  .then(res => console.log(`Asset was assigned an id of ${res.id}`));

.delete()

THIS METHOD HAS NOT BEEN FULLY DEVELOPED YET AND MAY NOT WORK

// For now, just an example of what it should look like eventually
farm.asset.delete(123);

Areas

An area is any well defined location that has been mapped in farmOS, such as a field, greenhouse, building, etc.

Here's an example of what an area looks like as a JavaScript object:

{
  tid: '22',
  name: 'F1',
  description: '',
  area_type: 'greenhouse',
  geofield: [
    {
      geom: 'POLYGON ((-75.53640916943549 42.54421203378203, -75.53607389330863 42.54421796218091, -75.53607121109961 42.54415472589722, -75.53640648722647 42.54414682135726, -75.53640916943549 42.54421203378203))',
    }
  ],
  vocabulary: {
    id: '2',
    resource: 'taxonomy_vocabulary'
  },
  parent: [
    {
      id: 11,
      resource: 'taxonomy_term'
    }
  ],
  weight: '0',
}

Methods for getting, sending and deleting areas are namespaced on the farm.area property.

.get()

Use the .get() method to retrieve a single area as a JavaScript object, or an array of objects, which can be filtered:

// Leave the parameter undefined to fetch all available areas
farm.area.get()
  .then(res => console.log(`Area #${res[0].tid} is called ${res.list[0].name}`))

// Accepts a number for the tid of the area you wish to fetch
farm.area.get(123)
  .then(res => console.log(`Area #123 is called ${res.name}`))

// Pass an options object to filter the results
farm.area.get({
  page: 2, // default === null
  type: 'field', // default === ''
}).then(res => console.log(`Area #${res[0].tid} is called ${res.list[0].name}`))

NOTE: Areas use a tid property, unlike logs and assets which have an id. This stands for taxonomy ID. In the future this may be changed to make it more consistent with the other entities.

The options object can have two properties: page is the page number in the sequence of paginated results, starting from 0 and in batches of 100 areas; type filters the results by area type.

Some common area types include:

  • field
  • building
  • property
  • water
  • other

Other area types may be provided by add-on modules in farmOS.

.send()

Send can be used to create a new area, or if the tid property is included, to update an existing area:

farm.area.send(area)
  .then(res => console.log(`Log was assigned an tid of ${res.tid}`));

.delete()

THIS METHOD HAS NOT BEEN FULLY DEVELOPED YET AND MAY NOT WORK

// For now, just an example of what it should look like eventually
farm.area.delete(123);

.geojson()

Get a GeoJSON document of all the farm's area geometries:

farm.area.geojson()
  .then(geojson => console.log(JSON.stringify(geojson)));

No parameters are accepted as of now, but in the future we hope to permit filtering by area type.

Taxonomy Terms

farmOS allows farmers to build vocabularies of terms for various categorization purposes. These are referred to as "taxonomies" in farmOS (and Drupal), although "vocabulary" is sometimes used interchangeably.

Some things that are represented as taxonomy terms include quantity units, crops/varieties, animal species/breeds, input materials, and log categories. See "Endpoints" above for specific API endpoints URLs.

A very basic taxonomy term JSON structure looks like this:

{
  "tid": "3",
  "name": "Cabbage",
  "description": "",
  "vocabulary": {
    "id": "7",
    "resource": "taxonomy_vocabulary",
  },
  "parent": [
    {
      "id": "10",
      "resource": "taxonomy_term",
    },
  ],
  "weight": "5",
}

The tid is the unique ID of the term (database primary key). When creating a new term, the only required fields are name and vocabulary. The vocabulary is an ID that corresponds to the specific vocabulary the term will be a part of (eg: quantity units, crops/varieties, log categories, etc). The fields parent and weight control term hierarchy and ordering (a heavier weight will sort it lower in the list).

.get()

Use the .get() method to retrieve a single vocabulary as a JavaScript object by passing in that vocabulary's machine name, or an array of objects, which can be filtered:

farm.term.get()
  .then(res => console.log(`Term #${res.list[0].tid} is called ${res.list[0].name}`))

farm.term.get('farm_crops')
  .then(res => console.log(`${res.list[0].name} is a crop.`))

farm.term.get({
  vocabulary: 'farm_crops',
  name: 'Icicle Radish',
  page: 0,
}).then(res => console.log(`It is ${(res.list.length > 0)} that 'Icicle Radish' is the name of a crop.`))

.send()

Send can be used to create a new taxonomy term, or if the tid property is included in the term object, to update an existing area:

farm.area.send(term)
  .then(res => console.log(`Log was assigned an tid of ${res.tid}`));

.delete()

THIS METHOD HAS NOT BEEN DEVELOPED YET

Vocabulary

Every taxonomy term is a member of a particular "vocabulary", so for instance, "Carrots" might be a taxonomy term, which is a part of the farm_crops vocabulary, while "Greenhouse #5" might be term in the farm_areas vocabulary. The vocabulary id, vid, will be a required field when creating a new term or updating an existing one, so in most cases it will be necessary to lookup the vid for the particular vocabulary to which it belongs before sending.

Here are the machine names of the more common vocabularies which are used in farmOS:

  • farm_animal_types
  • farm_areas
  • farm_crops
  • farm_crop_families
  • farm_log_categories
  • farm_materials
  • farm_quantity_units
  • farm_season
  • farm_soil_names

Note that the vid for each vocabulary may vary between different farmOS servers, so even if you're storing the vid in your application for one farmOS instance, you'll need to repeat the lookup process for any other instance to guarantee you have the proper vid.

The only method available is a getter for basic lookup:

// Get all vocabularies back in a list, by passing no argument
farm.vocabulary();

// Get a specific vocabulary by passing its machine name
farm.vocabulary('farm_crops');

It is not possible to create or modify a vocabulary via the API.

Farm & User Information

For requesting information about the farm, there is just one method, .info(), which is a getter.

.info()

farm.info()
  .then(res => console.log(`The farm's name is ${res.name}`))

MAINTAINERS

Current maintainers:

This project has been sponsored by:

About

A JavaScript library for interacting with farmOS over API.

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • JavaScript 100.0%