Kickserv APIv2 - Other Endpoints

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

/invoices.json

GET

GET paginated invoices. Includes a ‘total_balance_remaining’ key giving a total amount owed across all invoices. Every entry includes a customer glue record and task array.

Ex. GET /invoices.json retrieves the first page of invoices for the account.

Ex. GET /invoices.json?invoice_status=paid retrieves the first page of paid invoices for the account.

Ex. GET /invoices.json?invoice_status=unpaid&page=2 retrieves the second page of unpaid invoices for the account.

There’s no readily apparent way to retrieve a specific invoice, such as /invoices/{id}.json, ?invoice_id={id}, etc.

/opportunities.json

GET POST

GET paginated opportunities. The object format is similar to jobs and also returns a customer glue record.

Ex. GET /opportunities.json retrieves the first page of opportunities for the account.

Ex. GET /opportunities.json?status=new retrieves the first page of opportunities in ‘new’ status for the account.

Ex. GET /opportunities.json?job_type_id=24680&page=2 retrieves the second page of opportunities with service 24680 for the account.

Version 2 of the API doesn’t seem to contain a method for pulling opportunities for a specific customer only. ?customer_number= behaves like a valid parameter but testing it wasn’t successful. Version 3 contains an endpoint for doing this but it is not discussed here. While POST would be expected to function, the best way to create an opportunity (and the one recommended by Kickserv) to create a job, capture the returned job number, and then PUT the status to 'new' using /jobs/{id}.json.

New!

/opportunities/{id}/mark_as_lost.json

PUT

PUT an opportunity into 'lost' state, and return the opportunity.

Ex. PUT /opportunities/12345/mark_as_lost.json marks opportunity number 12345 as 'lost'.

Note that while some URLs can be either /opportunities/{id}.json or /jobs/{id}.json, this endpoint only exists as /opportunities
New!

/opportunities/{id}/mark_as_won.json

PUT

PUT an opportunity into 'won' state, and return the opportunity.

Ex. PUT /opportunities/12345/mark_as_won.json marks opportunity number 12345 as 'won'.

Note that while some URLs can be either /opportunities/{id}.json or /jobs/{id}.json, this endpoint only exists as /opportunities

/employees.json

GET POST

GET an array of active employees.

Ex. GET /employees.json returns an array of all active employees. Each entry will be similar to:

{
  "id": 846787,
  "account_id": 222222,
  "employee_number": 17,
  "name": "User Name",
  "admin": false,
  "active": true,
  "limit_access_to_assignments": false,
  "email_address": "user@somedomain.com",
  "username": "someusername",
  "address": "",
  "city": "",
  "state": "",
  "zip_code": "12345",
  "avatar": "https://someurl.png"
}  

While not exhaustively tested, initial attempts to create a new employee were not successful. Version 3 has not been tested. If there is a way to pull inactive employees, it hasn’t been identified. For reasons unknown, the returned record does not contain the phone number associated with an employee. That information is available in APIv3, but APIv3 doesn’t have an identified method for only retrieving one employee.

/employees/{id}.json

GET PUT DELETE

GET an employee as defined by the supplied employee number.

Ex. GET /employees/6.json returns employee number 6. The format is identical to each entry in the array returned by /employees.json:

{
  "id": 846787,
  "account_id": 222222,
  "employee_number": 17,
  "name": "User Name",
  "admin": false,
  "active": true,
  "limit_access_to_assignments": false,
  "email_address": "user@somedomain.com",
  "username": "someusername",
  "address": "",
  "city": "",
  "state": "",
  "zip_code": "12345",
  "avatar": "https://someurl.png"
}   

DELETE is not supported (even in the web UI) as that would potentially break the linkage for historical log entries ("Harry marked job 12345 complete"). Employees can only be marked 'inactive' but testing showed editing an employee is only possible via the web UI.
Returned records do not contain the phone number of the employee. That information is available in APIv3, but APIv3 doesn’t have an identified method for only retrieving one employee.

/vendors.json

GET POST

GET paginated vendors for the account.

POST a new vendor.

Ex. GET /vendors.json retrieves paginated vendors for the account. An example entry:

{
  "id": 555666777,
  "account_id": 112233,
  "name": "Wingus, Dingus & Doofus Attorneys at Law",
  "active": true,
  "created_at": "2020-12-23T06:27:50-07:00",
  "updated_at": "2020-12-23T06:27:50-07:00",
  "list_id": null
} 

Ex. POST {name:"Herp & Derp Accountants"} /vendors.json creates a new vendor with the supplied name. The new vendor object is returned. It contains an 'errors' key, which, if the name has already been used, will contain 'Name has already been taken'.


/vendors/{id}.json

GET PUT DELETE

PUT a change to an existing vendor, as defined by the supplied vendor ID.

Ex. PUT {name:"Hyphen and Comma, Punctuation Suppliers"} /vendors/159753.json changes the name of vendor 159753. The updated vendor object is returned. It contains an ‘errors’ key in case there was a problem with the supplied parameter.


/items.json

GET POST

GET paginated items for the account. A method for pulling a single item by ID has not been found. Items can be retrieved by name or partial name using the ?name= parameter; see the second example below.

Ex. GET /items.json returns the first page of items associated with the account. An example object:

{
  "id": 9955112233,
  "account_id": 335577,
  "created_on": "2019-12-19T05:55:46-07:00",
  "updated_on": "2021-02-25T07:28:29-07:00",
  "parent_list_id": null,
  "list_id": null,
  "edit_sequence": null,
  "name": "Widget",
  "sublevel": 0,
  "item_type": "",
  "price": "1.45",
  "active": true,
  "sales_tax_code_ref_full_name": null,
  "full_name": "Widget",
  "description": "A widget of some sort.",
  "sales_tax_code_ref_list_id": null,
  "tax_rate": null,
  "discount_rate": null,
  "discount_rate_percent": null,
  "taxable": false,
  "qb_error": null,
  "sales_tax_code_id": null,
  "updated_at": "2021-02-25T07:28:29-07:00",
  "formatted_price": "$1.45"
}

Ex. GET /items.json?name=Vio will return all items (paginated) whose name contains 'vio' (not case-sensitive). The records returned are truncated, containing only the fields outlined below. An example return:

"items": [
    {
      "id": 2536840,
      "name": "Violin Maintenance",
      "price": "55.0",
      "description": "Preventative maintenance on violin",
      "taxable": true,
      "formatted_price": "$55.00"
    }
],
"current_page": 1,
"per_page": 25,
"total_entries": 1,
"total_pages": 1,
"next_page": null,
"previous_page": null

POSTing returned a 404 in testing, regardless of whether or not the account was set to sync with Quickbooks. PUT did not function in testing for version 2 or 3, and it is not clear if it's even possible to create, edit, or delete an item using the API.

/schedule.json

GET

GET paginated tasks for the account. Tasks generally contain glue records for the job, customer, primary contact (if defined for the customer), and employees assigned to the task. See /jobs/{id}/tasks/{id}.json, /jobs/{id}/tasks.json, /tasks.json, and /tasks/{id}.json for additional information.

Ex. GET /schedule.json retrieves paginated tasks for the account.

Ex. GET /schedule.json?per_page=5&page=2 retrieves the second page of tasks, if a page is treated as 5 tasks.

Ex. GET /schedule.json?employee_number=4&task_type_id=2222 retrieves all tasks for employee number 4 of type 2222.


/payments.json

GET

GET paginated payments for the account. See /jobs/{id}/payments.json for additional information on record structure.

The pagination information for GET requests follows the version 3 model:

"meta": {
    "pagination": {
      "current_page": 1,
      "next_page": null,
      "prev_page": null,
      "total_pages": 1,
      "total_count": 7
    }
}

Ex. GET /payments.json retrieves paginated payments for the account.

Ex. GET /payments.json?per_page=5&page=2 retrieves the second page of payments, if a page is treated as 5 tasks.

Ex. GET /payments.json?customer_id=2673 retrieves paginated payments for customer 2673. Note that despite the parameter name, the supplied value is actually the customer number, not the ID.

Ex. GET /payments.json?customer_id=2673&sort_by=created_at&sort_order=DESC retrieves paginated payments for customer 2673 sorted by payment creation date (some other fields are also 'sortable') in descending order ('ASC' is also a valid value for sort_order).

POST and PUT have not been tested fully on this endpoint. POST does allow creation of a payment, but tying it to a specific job requires more than what's supplied in the payload.

/job_types.json

GET POST

GET paginated job types (called ‘Service’ in the web UI) for the account.

Ex. GET /job_types.json retrieves paginated job types for the account. An example object, contained inside a job_types array:

{
  "id": 9948757,
  "account_id": 12345,
  "name": "Yodeling Lessons",
  "color": "000",
  "position": 99,
  "active": true,
  "created_at": "2020-06-26T11:46:30-04:00",
  "updated_at": "2020-06-26T11:46:30-04:00",
  "description": "",
  "cost": "",
  "time": 0,
  "text_color": "fff",
  "public": false
}    

Ex. GET /job_types.json?inactive=true returns paginated inactive job_types (defaults to only returning active job_types).

Ex. GET /job_types.json?all=true returns paginated active and inactive job_types.

?active=true is a valid parameter, but it is also the default. There does not appear to be a method for retrieving only a single job_type by ID, though it is sometimes returned by other endpoints as a glue record.

/job_charge_types.json

GET

GET paginated job_charge_types (for example, a charge is 'Hourly', a 'Physical Product', etc.

Ex. GET /job_charge_types.json retrieves the first page of job_charge_types.

{
  "job_charge_types": [
    {
      "id": 12345,
      "account_id": 11111,
      "name": "Hourly",
      "active": true,
      "created_at": "2020-03-03T16:46:15-05:00",
      "updated_at": "2020-03-03T16:46:15-05:00"
    },
    {
      "id": 12346,
      "account_id": 11111,
      "name": "Non-recurring",
      "active": true,
      "created_at": "2020-03-04T14:59:47-05:00",
      "updated_at": "2020-03-04T14:59:47-05:00"
    }
  ],
  "current_page": 1,
  "per_page": 25,
  "total_entries": 5,
  "total_pages": 1,
  "next_page": null,
  "previous_page": null
}


While most endpoints follow the pattern of https://app.kickserv.com/SLUG/api/v2/ENDPOINT.json there are some that for whatever reason do not. It is not known if they are considered by Kickserv to be part of APIv2 or if they are vestigial remnants of an earlier (or potentially, later) version of the API. These are not described here in as detailed a fashion, but merely for reference. There are plenty of other endpoints that are used by the web UI but typically the payload is accompanied by a nonce/authenticity token. As it’s not known how these tokens are generated, those endpoints are not listed here at all.

Note that these all have the form /SLUG/PATH/TO/ENDPOINT.json and do not contain /api/v2. They have been tested with the GET verb only unless otherwise indicated.

/SLUG/tags/autosuggest.json

POST - Used to look up tags. Takes two parameters, 'query' and 'type'. The string in 'query' will be used to look up tags containing that string. Valid values for type are 'Job' and 'Customer'. An empty string in query will bring back a full list for that type. The only other way to get a list of tags is via the /SLUG/calendar.json endpoint, but the array it returns does not contain the tag IDs.

/SLUG/dashboard/dashboard_info.json

Returns information used to generate the dashboard that is usually seen when first logging in. There is one known parameter that can take one of several values. ?get_info can take one of the following: 'jobs_info', 'opportunities_info', 'invoices_info', or 'tasks_info'.

/SLUG/calendar.json

Contains employees, service IDs, task type IDs, tag names, statuses. Does not contain tag IDs.

/SLUG/calendar

Contains employees, service IDs, task type IDs, tag names, statuses. Does not contain tag IDs.

/SLUG/calendar/events.json

Not heavily researched, related to Dispatch screen. Accepts ?timeshift=, ?uid=, ?from=, and ?to=.

/SLUG/schedule.json

Always seems to return an empty array, but presumably it has some function. It may require certain parameters to return any data.

/SLUG/employees.json

Returns a shortened version of the /employees.json endpoint.

/SLUG/jobs.json

Returns totals for unscheduled, scheduled, in progress, on hold, and complete jobs as a simple object, including value totals.

/SLUG/jobs

Returns totals for unscheduled, scheduled, in progress, on hold, and complete jobs as a simple object, including value totals.

/SLUG/opportunities.json

Returns totals for new, scheduled, sent, viewed, and lost opportunities as a simple object including value totals.

/SLUG/opportunities

Returns totals for new, scheduled, sent, viewed, and lost opportunities as a simple object including value totals.

/SLUG/contacts.json

Returns all contacts, without pagination.

/SLUG/contacts

Returns all contacts, without pagination.

/SLUG/customers/{id}/payments.json

Returns payments.

/SLUG/customers/{id}/payments/{id}.json

Returns a specific payment as defined by the supplied customer number and payment ID.

/SLUG/map/checkins.json

Returns checkins from mobile app. ?date=2021-08-13&employee_id=1234567

/SLUG/employees/{id}/employee_settings.json

Returns ordering information for the given employee number, used in web UI.

/SLUG/employees/unavailable_hours.json?day_of_week={number}

Returns the hours that each employee is unavailable for the given day (0-6, Monday is 1).

/SLUG/job_types/{id}.json

Returns information about the supplied job_type id.