Homepage

Background Jobs

Last edit: May 25, 2024

In this article, we share our experience on how you can improve your application performance by utilizing our background Liquid tag.

Synchronous vs asynchronous

To explain the synchronous and asynchronous concepts of code, we take a definition from a school, because we think it explains this concept in an easier to understand way.

Synchronous

Synchronous learning is when classes occur on set schedules and time frames. Students and instructors are online at the same time in synchronous classes since lectures, discussions, and presentations take place at specific hours. All students must be online at that exact time in order to participate in the class.

Source: https://www.elearners.com

In terms of code, it means that operations are executed one by one. Every task needs to finish, before the next one is executed.

const log = msg => console.log(msg);

log('Hello');
log('World');

This code will log messages in the same order as they appear on the screen:

Hello
World

Asynchronous

Asynchronous classes let students complete their work on their own time. Students are given a timeframe – it's usually a one-week window – during which they need to connect to their class at least once or twice. The good news is that in asynchronous courses, you could hit the books no matter what hour of day (or night).

Source: https://www.elearners.com

In terms of code, it means that operations are pushed to a different context and the program executes onward, without waiting for them to finish.

const log = msg => console.log(msg);

log('Start');
setTimeout(() => { log('Hello'); }, 5000);
setTimeout(() => { log('World'); }, 1000);
log('Finish');

This piece of code will log:

  1. Start - Immediately
  2. Finish - Immediately
  3. World - Delayed by one second
  4. Hello - Delayed by five seconds

Tasks in setTimeout functions are run asynchronously. We know it is asynchronous, because the order of messages logged is not the same as it's defined in code, but are based on when the tasks actually finished.

Asynchronous code is usually used to maximize the efficiency of CPU cycles, minimizing block time of the main thread, and wait time for situations where those operations can be potentially long-running.

Why background jobs

In platformOS, the background tag allows you to push some operations to the asynchronous queue that will be run independently of the currently served http request. This means it will not block server response to the user.
Background jobs should be used as performance improvement everytime the work is time consuming (subjectively, we usually try to push any work over 1 second to a background job).

Priorities

You can choose a priority for your job based on its time-sensitivity and estimated time of computation needed. All of them have different timeouts after the job will be killed no matter if it has finished or not.

Priority Timeout
high 1 minute
default 5 minutes
low 1 hour
  • If you need something to happen as soon as possible, choose the high priority.
  • If you need more than one minute of computation time, or there is a risk of exceeding 1 minute, choose the default priority.
  • If it's not a time-sensitive job, or it is a long-running job, choose the low priority.

Execution scope


{% background priority: 'low', delay: 5, max_attempts: 3, source_name: "my custom job", data: my_data, hey: 'hello' %}
{% endbackground %}

The background tag creates a completely new execution context, so you will only have access to variables explicitly provided to the background tag: "data" and "hey". Note that you will not have access to the "my_data" variable. Also note
that the result of this background tag will not be rendered, and you will not have access to any variable inside the background tag.

Example 1 - high priority job

Download a JSON file and put its value into the record field. In this example, we will download the JSON file with currency rates for USD and put the resulted JSON into a database to use it further down the road.



{% assign currency = context.params.currency %}

{% background priority: 'high', source_name: "update-currency-rates", currency: currency %}
  {% capture url %}https://example.com/currency-rates.json?currency={{ currency }}{% endcapture %}

  {% assign data = url | download_file | parse_json %}

  {% graphql g, currency: "USD", rates: data %}
    mutation update_currency($currency: String!, $rates: String!) {
      record_update(
        id: 1337
        record: {
          table: $currency
          properties: [{ name: "rates", value: $rates }]
        }
      ) {
        id
      }
    }
  {% endgraphql %}
{% endbackground %}

This whole operation, because it was pushed out to background job, will not block rendering of the page, hence standard request timeout is not applied to it. It can run as long as the timeout of this particular priority - 1 minute.

Example 2 - normal priority job

In this case, we will update one field (price) in all 500 products in an e-commerce store. In a real application, discount would probably be a separate field but for the purpose of this example let's assume there is no such field.



{% background priority: 'normal', source_name: 'apply-discount' %}
  {% graphql products %}
    query products {
      records(
        per_page: 1000
        filter: {
          table: { exact: "products" }
        }
      ) {
        results {
          id
          price: property_int(name: "price_cents")
        }
      }
    }
  {% endgraphql %}

  {% assign prods = products | fetch: "records" | fetch: "results" %}

  {% for prod in prods %}
    {% comment %}Apply 15% discount{% endcomment %}

    {% assign newPrice = prod.price | times: 0.85 | floor %}

    {% graphql updated, prod_id: prod.id, new_price: newPrice %}
      mutation update_prices($new_price: Int!, $prod_id: Int!) {
        record_update(
          id: $prod_id
          record: {
            table: "products"
            properties: [{ name: "price_cents", value: $new_price }]
          }
        ) {
          id
        }
      }
    {% endgraphql %}
  {% endfor %}

{% endbackground %}

This operation probably will not take more than one minute, but because this is not a time-sensitive task, we will let it run in the normal priority queue.

Example 3 - long-running job

When you need to do something that is very time-consuming, use the low priority. We don't have any use case for that yet, but maybe you have - we got you covered.


{% background priority: 'low', max_attempts: 3, source_name: "long-running-job %}
  {% comment %}
    Code
  {% endcomment %}
{% endbackground %}

Questions?

We are always happy to help with any questions you may have.

contact us