This documentation is neither written nor endorsed by Kickserv. Use at your own risk.

/jobs.json

GET POST

GET an array of paginated jobs or POST a new one.

GET defaults to 25 jobs per page and without any other parameters starts with the most recent job, descending by job number.

POST requires two parameters, 'job_type_id' and 'customer_number'. Other fields may be supplied as desired. Early testing indicates that charges can not be supplied while creating a job. They must be added in a separate call; see /jobs/{id}.json or /jobs/{id}/job_charges.json.

See /customers/{id}/jobs.json for another way to create jobs. No difference between the two endpoints has been found. Note the format of a POST payload in the example below. A shortened version of active jobs (not canceled, lost, complete) list containing only the ID, name, and status is available at /tasks/job_list.json.

It seems that charges can not be included in the initial creation of a job. When a job is created, extract the new job number from the return value and then POST to /jobs/{id}/job_charges.json or PUT to /jobs/{id}.json. The exact payload will vary depending on which of these endpoints is used. /customers/{id}/jobs/{id}.json has not been tested for adding or modifying charges but is not expected to work.
POST will create a job, not an opportunity. One of the handful of Kickserv documents that mentions the API confirms that the preferred way to create an opportunity is to create a job, and then modify its status to 'new'.

Ex. GET /jobs.json returns the first page of all jobs under the account starting with the most recent.

Ex. GET /jobs.json?per_page=30&page=2 returns the second page of jobs for the account, including 30 records per page as opposed to the default of 25.

Ex. GET /jobs.json?job_type_id=44111&scheduled=today retrieves the first page of jobs scheduled for today of type 44111. Other valid 'scheduled' values are 'tomorrow', 'this_week', 'next_week', 'last_week', 'this_month', 'next_month', and 'last_month'.

Ex. POST {job:{job_type_id:24551, description:"A new job from the API!"},customer_number:8} /jobs.json creates a new job of type 24551 (the ID of a service) with the listed description for customer number 8. The newly-created job is returned.


/jobs/{id}.json

GET PUT DELETE

GET a specific job or opportunity as defined by the supplied job number.

PUT changes to a specific job or opportunity as defined by the supplied job number. Multiple changes can be made in a single call if they are included in the payload. For example: {job:{description:"A new description", scope_of_work:"An updated scope"}}. With the exception of keys that update as a result of the query, no keys will change. That means that if you update a job's description the scope of work won't change and doesn't have to be included in each query. However, there are some keys (such as 'updated_by', 'updated_at', etc) that will updated whether you want them to or not. As a general rule keys like those are not directly editable.

DELETE a specific job or opportunity as defined by the supplied job number.

Ex. GET /jobs/456.json returns job 456.

Ex. PUT {job:{description:"A new description"}} /jobs/456.json modifies the description of job number 456. Returns an updated job object, but custom field information is not included.

Ex. DELETE /jobs/456.json will delete the job, and return it.

GET and PUT return objects with slightly different formats.

/jobs/{id}/notes.json

GET POST

GET an array of notes from a specific job or opportunity as defined by the supplied job number. Note objects at the job and customer level are indistinguishable except for one key, the noteable_type key. That key will be either 'Customer' or 'Job'. For customer-level notes, see /customers/{id}/notes.json.

POST a note to a specific job or opportunity as defined by the supplied job number.

Ex. GET /jobs/1045/notes.json returns notes for job 1045

Ex. POST {note:{note:"A new description", public:false}} /jobs/1045/notes.json adds a new private note to job number 1045. Returns the newly-created note.

GET and PUT return objects with slightly different formats.

/jobs/{id}/notes/{id}.json

GET PUT DELETE

GET a specific note for a specific job or opportunity as defined by the supplied job number and note ID. Note IDs are not easily visible in the web UI so the best way to obtain them is via the return from an endpoint like /jobs/{id}/notes.json.

PUT a change to a specific note for a specific job or opportunity as defined by the supplied job number and note ID.

DELETE a specific note for a specific job or opportunity as defined by the supplied job number and note ID.

Ex. GET /jobs/631/notes/6842.json retrieves job 6842 from job 631.

Ex. PUT {note:{note:"New note body"}} /jobs/631/notes/6842.json modifies the body of note 6842 from job 631 and returns the modified note.

Ex. DELETE /jobs/631/notes/6842.json deletes note 6842 from job 631 and returns the note.

Testing showed that the important part of the endpoint is the note ID. While the job number should be accurate, it does not appear to be validated (e.g., note 6842 would still be deleted even if it wasn’t associated with job 631).
If a note is edited, the 'public' key is reset to 'false' if it is not provided in the payload. If editing a note that’s visible to the customer, make sure to include the 'public' key set to 'true' or it will no longer be visible to them.

/jobs/{id}/time_entries.json

GET POST

GET all time entries (paginated) for a specific job or opportunity as defined by the supplied job number.

Ex. GET /jobs/4512/time_entries.json retrieves time entries from job 4512.

One would expect this to be a POST-able endpoint but in APIv2 that does not appear to be the case. This is the only way to retrieve time_entries as a separate object found to date. No API endpoint allowing you to GET a specific time entry (like /jobs/{id}/time_entries/{id}.json) has been identified. As such, there's also no known mechanism for modifying or deleting an existing time entry, as those two operations would likely be found on the same endpoint.

/jobs/{id}/job_charges.json

GET POST

GET all charges (paginated) for a specific job or opportunity as defined by the supplied job number. Jobs are returned inside a 'job_charges' object.

POST a new charge to an specific job or opportunity as defined by the supplied job number. This is not a wipe-and-replace operation - the new charge will be added to any existing charges. The new charge is returned. item_id is the only key technically required to create a new charge, but see the note below.

An example charge object:

{
  "id": 7777777,
  "job_id": 88888888,
  "account_id": 70023,
  "created_on": "2020-03-19T13:59:47-04:00",
  "description": "Costume, Gorilla",
  "quantity": "2.0",
  "price_per_unit": "50.00",
  "item_id": 92929292,
  "qb_txn_line_id": null,
  "details": "A costume which resembles a gorilla",
  "item_ref_list_id": null,
  "txn_line_id": null,
  "total": "100.00",
  "job_charge_type_id": null,
  "updated_by": 565656,
  "recurring_item_parent_id": null,
  "deleted_at": null,
  "tax_id": null,
  "sales_tax_code_id": null,
  "discount": "0.0",
  "updated_at": "2020-03-19T13:59:47-04:00",
  "sort_order": null,
  "line_total": "100.00"
}
                        

Notice that the charge object contains the employee ID of the last employee to make a change to it, stored in the updated_by field.

Ex. GET /jobs/11325/job_charges.json retrieves charges from job 11325

Ex. POST {job_charges:{item_id:24601, quantity:5, description:"Chips, Potato"}} /jobs/11325/job_charges.json adds 5 potato chips (item number 24601) to job 11325

If item_id is the only supplied key in a POST operation the price will be $0.00 and the details will be blank. It is recommended to always include those two keys when creating a new charge as they don’t auto-populate like they do in the web UI. If a job is locked because it’s been synced to Quickbooks (or presumably Xero, but I have no personal experience) an error will be returned if you attempt to POST a new charge. Trailing spaces will be trimmed from the 'description' key upon submission.

If job charges are examined from the '.job_charges' key in a reply from the /jobs/{id}.json endpoint, a glue record for the item is returned as well. The charge contains a .price_per_unit key and in most cases the .item.price will be the same number. However, if a charge is added using the discount flag or the price is manually changed then .item.price will be the item default price and will NOT match .price_per_unit.
There are two fields that are easy to confuse, 'details' and 'description'. 'Description' is what’s seen in the 'Item' column in the web UI and 'details', confusingly enough, is what's in the 'Description' field in the web UI.

/jobs/{id}/job_charges/{id}.json

GET PUT DELETE

GET a specific charge for a specific job or opportunity as defined by the supplied job number and charge ID.

PUT a change to a specific charge for a specific job or opportunity as defined by the supplied job number and charge ID.

DELETE a specific charge for a specific job or opportunity as defined by the supplied job number and charge ID. Unlike some other endpoints, deleting a charge simply returns a success message instead of the charge that was deleted.

Ex. GET /jobs/6517/job_charges/69697.json retrieves charge 69697 from job 6517

Ex. PUT {job_charge:{quantity:3}} /jobs/6517/job_charges/69697.json changes the quantity of charge 69697 to 3

Ex. DELETE /jobs/6517/job_charges/69697.json deletes charge 69697 from job 6517


/jobs/{id}/tasks.json

GET POST

GET always returns an empty array, but presumably it can take some parameters that make it return something useful. None have yet been identified. To retrieve tasks for a job or opportunity see /tasks.json.

POST a new task to an existing job or opportunity. This can be used to create a task of any type simply by changing the task_type_id value in the payload.

DELETE a specific charge for a specific job or opportunity as defined by the supplied job number and charge ID. Unlike some other endpoints, deleting a charge simply returns a success message instead of the charge that was deleted.

Ex. GET /jobs/4478/tasks.json retrieves an empty array if no additional parameters are provided.

Ex. POST (see payload below) /jobs/4478/tasks.json will create a new task of type SOME-NUMERIC-TASKTYPE for job 4478.

{
    "task": {
        "name": "On-site estimate",
        "description": "...and some smaller text",
        "job_id": 1122334455,
        "scheduled_at": "2021-05-28T08:00:00-04:00",
        "duration": 60,
        "ends_at": "2021-05-28T09:00:00-04:00",
        "task_type_id": SOME-NUMERIC-TASKTYPE,
        "employee_ids":[EMPLOYEE-ID,OTHER-EMPLOYEE-ID]
    }
}
                        

See Common Fields and Where to Find Them for a discussion of task_type_id. The 'employee_ids' field is a comma-separated array of the employee IDs of all the employees you want assigned to this task. Even if there’s only one, it still needs to be in an array.

Do not confuse the 'employee_id' (singular) field with the 'employee_ids' (plural) field. The 'employee_id' field (singular) does not appear to be used and is presumably a vestigial holdover from a previous implementation of task assignment. 'employee_ids' should always be an array even if there's only a single value in it.

The job_id field should be derivable (since you're POSTing to a URL containing the job number) however this field is still required. As such, there’s no obvious benefit to using this endpoint over /tasks.json. There's been no functional difference identified.

While the fields 'duration', 'ends_at', and 'scheduled_at' are not required for the task to be created they are not automatically populated (for example, a 'scheduled_at' and 'duration' will not result in 'ends_at' being calculated). The behavior of Kickserv for things like automatic reminders of a partially-assembled task has not been tested, so it’s better to just treat those fields as if they were required.

'name' maps to what's called 'description' in the web UI and is the text field normally used for adding a note about the task. 'description’ produces some slightly smaller text below the 'name' text in the web UI and is not editable via the web UI. This can be a handy way of capturing and displaying who created a task since the only way to change it is to delete the entire task (which can be controlled with permissions). 'name' should never be null as it will cause the task creation to fail if the key is present. If you're not using it, use an empty string for the value.


/jobs/{id}/tasks/{id}.json

GET PUT DELETE

GET a specific task for a specific job or opportunity as defined by the supplied job number and task ID. Tasks generally contain glue records for the job, customer, primary contact (if defined for the customer), and employees assigned to the task.

PUT a change to a specific task for a specific job or opportunity as defined by the supplied job number and charge ID.

DELETE a specific task for a specific job or opportunity as defined by the supplied job number and task ID.

Ex. GET /jobs/6517/tasks/69697.json retrieves task 69697 from job 6517.

Ex. PUT {task:{name:"A new task name"}} /jobs/6517/tasks/69697.json changes the 'name' field of task 69697.

Ex. DELETE /jobs/6517/tasks/69697.json deletes task 69697 from job 6517 and returns it.


/jobs/{id}/payments.json

GET POST

GET paginated payments for a specific job, as defined by the supplied job number. The pagination data returned is in the format of APIv3, meaning that it’s contained as follows:

{
  "payments": [ ...payment objects... ],
  "meta": {
    "pagination": {
      "current_page": 1,
      "next_page": 2,
      "prev_page": null,
      "total_pages": 2,
      "total_count": 26
    }
  }
}
                        

A single payment object looks like this:

    {
      "id": 66554433,
      "created_at": "2021-03-02T11:22:57-05:00",
      "updated_at": "2021-03-02T15:24:24-05:00",
      "payment_type": "Credit Card",
      "amount": "168.22",
      "notes": null,
      "ref_number": null,
      "synced": true,
      "linked_payment_jobs": "<a href=\"/YOURSLUG/jobs/JOBNUMBER\">JOBNUMBER</a>",
      "formatted_amount": "$168.22",
      "job_numbers": [
        JOBNUMBER
      ],
      "date": "2021-03-02",
      "payment_application_ids": [
        33557332
      ],
      "customer_payment_url": "/YOURSLUG/customers/CUSTOMERNUMBER/payments/66554433",
      "edit_customer_payment_url": "/YOURSLUG/customers/CUSTOMERNUMBER/payments/66554433/edit",
      "xero_payment": null
    }
                        

POST a new payment for a specific job as defined by the supplied job number.

Ex. GET /jobs/4467/payments.json retrieves paginated payments for job 4467.

Ex. POST {amount:"77.00", account_id:112233, customer_id:7745443} /jobs/4467/payments.json will create a $77 payment for job 4467. This is the absolute minimum to have a payment display, but it will have a blank payment type and an incorrect date. Adding 'date' ('05/05/2021') and 'payment_type' ("Credit Card") are recommended. The oddity here is that 'customer_id' and 'account_id' (your customer's ID and the ID of your Kickserv account) are not listed in any payment object but are required to POST one. Your Kickserv account ID is visible as the ‘account_id’ field in many objects including any job returned from /jobs.json. The format of the objects returned from GET and a successful POST are not the same.

Methodical testing of all possibilities has not been performed on whether payments created using this endpoint sync properly to Quickbooks or Xero. That being said, code using this endpoint has been in production for over a year with no complaints from our accountants. Payments with all relevant information supplied sync without problems.

The structure of the payment object suggests that it's possible to apply a payment to more than one job, though that has not been tested.

/jobs/{id}/expenses.json

GET POST

GET expenses for the job or opportunity specified by the supplied job number.

POST an expense for the job or opportunity identified by the supplied job number.

DELETE a specific task for a specific job or opportunity as defined by the supplied job number and task ID.

Ex. GET /jobs/6967/expenses.json retrieves paginated expenses for job 6967.

Ex. POST (see payload below) /jobs/6967/expenses.json creates a new expense for job 6967. While the payload just specifies an employee_id and a vendor_id the returned object also contains glue records for the employee and vendor negating the need for queries to look them up. Required fields are employee_id, vendor_id, payment_type, and date. The value specified for payment_type is a string, and must be an exact case-sensitive match of an existing payment type.

{
  "expense": 
    {
      "employee_id": 4568922,
      "vendor_id": 1354546,
      "billable": false,
      "description": "Some expense",
      "po_number": "565656",
      "receipt_number": "54321",
      "amount": "15.75",
      "payment_type": "Cash (out of pocket)",
      "date": "2021-04-04"
    }
}
                        

While it’s possible in the web UI to have an expense be billable and be represented as a charge on the job the specifics of how to accomplish that via the API have not yet been identified. It may be that the API engine does not actually handle this in a single query, although a quick test indicates that attempting to map a new expense to an existing charge will fail.

/jobs/{id}/expenses/{id}.json

GET PUT DELETE

GET a specific expense for a specific job, as defined by the supplied job number and expense ID.

PUT a change to a specific expense for a specific job or opportunity as defined by the supplied job number and expense ID.

DELETE a specific expense for a specific job or opportunity as defined by the supplied job number and expense ID.

Ex. GET /jobs/6967/expenses/15995.json retrieves expense 15995 for job 6967.

Ex. PUT {po_number:23445} /jobs/6967/expenses/15995.json changes the PO number for expense 15995 on job 6967. This does not return any object.

Ex. DELETE /job/6967/expenses/15995.json deletes expense 15995 on job 6967 and returns an empty object.


/jobs/{id}/taggings.json

POST

POST a new set of tags to a job or opportunity, as defined by the supplied job or opportunity number.

Ex. POST {tags:[ "High priority", "Wholesale" ] } /jobs/2434/taggings.json adds the 'High priority' and 'Wholesale' tags to job 2434.

This is a wipe-and-replace operation. Supplied tags are NOT appended to tags that are already on a job. See /SLUG/tags/autosuggest.json in the Other Endpoints section.

Ready to move on? Up next are tasks