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
.
/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.
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.
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 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 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.
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.
/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.
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
/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
.
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
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.
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.
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.
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" } }
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.
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.
/SLUG/tags/autosuggest.json
in the Other Endpoints section.