The Mochi API is organized around REST. It accepts JSON or transit+json encoded request bodies and returns either json or transit+json encoded responses. When making requests with json, keyword parameters must be sent as strings.
Base URL:
https://app.mochi.cards/api/
Mochi uses API Keys to authenticate requests. You can view and manage your API Keys in the Account Settings window within the app.
Authentication to the API is performed via HTTP Basic Auth. Provide your API key as the basic auth username value. You do not need to provide a password.
curl https://app.mochi.cards/api/decks \
-u my_secret_key_goes_here:
# The colon prevents curl from asking for a password.
Some requests return a list of resources. The response will be an object with a docs
property and a bookmark
property. The docs
property will contain alist of the documents, and the bookmark property can be used in subsequent requests to fetch the next logical page of the collection. Not every request that returns a bookmark has additional pages.
{
"bookmark": "g1AAAABAeJzLYWBgYMpgSmHgKy5JLCrJTq2MT8lPzkzJBYpzZGdmBxqXeTuB5Dlg8igyWQA7BRJR",
"docs": [
{
"id": "HJBMBGBO",
"sort": 6,
"name": "New Deck"
},
...
]
}
Mochi uses conventional HTTP response codes to indicate the success or failure of an API request. In general: Codes in the 2xx range indicate success. Codes in the 4xx range indicate an error that failed given the information provided (e.g., a required parameter was omitted, a document was not found, etc.). Codes in the 5xx range indicate an error with Mochi's servers (these are rare).
In most cases the API will also return a JSON or Transit response with more information about the error. In the case of validation errors, there will be a key corresponding to each field that failed validation.
Example error responses:
500 { "errors" ["Something went wrong!"] }
403 { "errors" ["Please upgrade to use this feature."] }
422 { "errors" { "content": ":content field cannot be nil." } }
Endpoints:
GET /cards
POST /cards
GET /cards/id
POST /cards/id
DELETE /cards/id
Cards can be created one at a time. Cards can be stored in a deck and can contain markdown content.
content
requireddeck-id
requiredtemplate-id
optionalarchived?
optionalreview-reverse?
optionalpos
optional:pos
of "6"
and card B has a :pos
of "7"
, you can create a new card in between them by setting its :pos
to "6V"
.manual-tags
optional#
character.fields
optional<< Field name >>
. Each value in the map is another map with the following structure:id
requiredvalue
requiredExample request
POST https://app.mochi.cards/api/cards/
Content-Type: application/json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
{
"content": "New card from API. ![](@media/foobar03.png)",
"deck-id": "btmZUXWM",
"template-id": "8BtaEAXe",
"fields": {
"name": {
"id": "name",
"value": "Hello,"
},
"JNEnw1e7": {
"id": "JNEnw1e7",
"value": "World!"
},
},
"review-reverse?": false,
"archived?": false,
"pos": "6V",
"manual-tags": ["philosophy", "philosophy/aristotle"]
}
POST https://app.mochi.cards/api/cards/
Content-Type: application/transit+json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
{
"~:content": "New card from API. ![](@media/foobar03.png)",
"~:deck-id": "~:btmZUXWM",
"~:template-id": "~:8BtaEAXe",
"~:fields": {
"~:name": {
"~:id": "~:name",
"~:value": "Hello,"
},
"~:JNEnw1e7": {
"~:id": "~:JNEnw1e7",
"~:value": "World!"
},
},
"~:review-reverse?": false,
"~:archived?": false,
"~:pos": "6V",
"~:tags": {"~#set": ["philosophy", "philosophy/aristotle"]}
}
This will return the created card if successful. Will return an error if something goes wrong, usually if required parameters are missing, or if supplied parameters fails validation.
Retrieves a card that already exists. The returned data will look identical to the data returned from the card creation request.
No parameters.
Example request
GET https://app.mochi.cards/api/cards/btmZUXWM
Content-Type: application/json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
Example response
{
"updated-at": {
"date": "2021-09-11T14:23:53.250Z"
},
"tags": [],
"content": "New card from API.",
"name": null,
"deck-id": "btmZUXWM",
"pos": "00F",
"references": [],
"id": "QQJ8ssvL",
"reviews": [],
"created-at": {
"date": "2021-09-10T01:29:49.879Z"
},
"new?": false,
"archived?": true,
"template-id": null
}
{
"~:updated-at": {
"~#dt": 1631370233250
},
"~:tags": {
"~#set": []
},
"~:content": "New card from API.",
"~:name": null,
"~:deck-id": "~:btmZUXWM",
"~:pos": "00F",
"~:references": {
"~#set": []
},
"~:id": "~:QQJ8ssvL",
"~:reviews": [],
"~:created-at": {
"~#dt": 1631237389879
},
"~:new?": false,
"~:archived?": true,
"~:template-id": null
}
Lists cards in pages of 10 cards per page. Use the bookmark
attribute to request the next page. Optionally specific a deck-id
to fetch only cards for that deck.
deck-id
optional
string - Only return cards for the deck specified by this deck ID.
limit
optional
number - The number of cards to return per page. The limit can range from 1 and 100, and the default is 10.
bookmark
optional
string - A cursor for use in pagination. Returned from a previous list request that has additional documents.
Example request
# Without a bookmark
GET https://app.mochi.cards/api/cards/
# With a bookmark
GET https://app.mochi.cards/api/cards?bookmark=some-bookmark
# With a deck-id
GET https://app.mochi.cards/api/cards?deck-id=my-deck-id
A dictionary with a docs
property that contains an array of up to limit
cards, and a bookmark
property that can be used to fetch the next page of results.
Example response
{
"bookmark": "g1AAAABAeJzLYWBgYMpgSmHgKy5JLCrJTq2MT8lPzkzJBYpzGGY5BzimZYaD5Dlg8igyWQAw3xHv",
"docs": [
{
"tags": [],
"content": "# Hello, world!",
"name": "Hello world",
"deck-id": "eH53Hxe8",
"fields": {},
"pos": "1",
"references": [],
"id": "00UlY4dd",
"reviews": [],
"created-at": {
"date": "2021-09-09T02:49:58.535Z"
}
},
...
]
}
{
"~:bookmark": "g1AAAABAeJzLYWBgYMpgSmHgKy5JLCrJTq2MT8lPzkzJBYpzGGY5BzimZYaD5Dlg8igyWQAw3xHv",
"~:docs": [
{
"~:tags": {
"~#set": []
},
"~:content": "# Hello, world!",
"~:name": "Hello world",
"~:deck-id": "~:eH53Hxe8",
"~:fields": {},
"~:pos": "1",
"~:references": {
"~#set": []
},
"~:id": "~:00UlY4dd",
"~:reviews": [],
"~:created-at": {
"~#dt": 1631155798535
}
},
...
]
}
Updating a card accepts most of the same parameters as creating a card. The ID of the card must be passed in as part of the URL.
content
optionaldeck-id
optionaltemplate-id
optionalarchived?
optionaltrashed?
optionalreview-reverse?
optionalpos
optional:pos
of "6"
and card B has a :pos
of "7"
, you can create a new card in between them by setting its :pos
to "6V"
.manual-tags
optional#
character. Setting this will overwrite whatever value for this field was already set on the card.fields
optional<< Field name >>
. Each value in the map is another map with the following structure:id
requiredvalue
requiredExample request
POST https://app.mochi.cards/api/cards/00UlY4dd
Content-Type: application/json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
{
"content": "New card from API. ![](@media/foobar03.png)",
"deck-id": "btmZUXWM",
"review-reverse?": false,
"archived?": false,
"pos": "6V",
"manual-tags": ["philosophy", "philosophy/aristotle"]
}
POST https://app.mochi.cards/api/cards/00UlY4dd
Content-Type: application/transit+json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
{
"~:content": "New card from API. ![](@media/foobar03.png)",
"~:deck-id": "~:btmZUXWM",
"~:review-reverse?": false,
"~:archived?": false,
"~:pos": "6V",
"~:tags": {"~#set": ["philosophy", "philosophy/aristotle"]}
}
Permanently deletes a card and its attachments.
Warning
This action cannot be undone. For a "soft" deletion you can update a card and set its trashed?
property to the current time.
None.
Example request
DELETE https://app.mochi.cards/api/cards/00UlY4dd
Content-Type: application/json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
DELETE https://app.mochi.cards/api/cards/00UlY4dd
Content-Type: application/transit+json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
You can add an attachment to a card using a separate request. This request must be sent as form data with the content-type of form/multi-part
.
None.
Example request
POST https://app.mochi.cards/api/cards/00UlY4dd/attachments/Q9DlrByP.mp3
Content-Type: multipart/form-data; boundary=------------------------924hzYpC8yIyyc579xl9Zw
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
POST https://app.mochi.cards/api/cards/00UlY4dd/attachments/Q9DlrByP.mp3
Content-Type: multipart/form-data; boundary=------------------------924hzYpC8yIyyc579xl9Zw
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
Example curl request
curl -POST https://app.mochi.cards/api/cards/00UlY4dd/attachments/Q9DlrByP.mp3 -u my_api_key: \
-H "Content-Type: multipart/form-data" \
-F file="@/path/to/attachment/my_recording.mp3"
Example fetch request
const formData = new FormData();
formData.append("file", someBlob, "Q9DlrByP.mp3");
fetch("https://app.mochi.cards/api/cards/00UlY4dd/attachments/Q9DlrByP.mp3", {
method: "POST",
body formData
});
Example python request
imoprt requests
def add_attachment(file_name, card_id):
url = f"https://app.mochi.cards/api/cards/{card_id}/attachments/{file_name}"
basic = HTTPBasicAuth(api_key, "")
with open(file_name, 'rb') as audio_file:
data = {
'file': (file_name, audio_file, 'audio/wav')
}
requests.post(url, files=data, auth=basic)
Decks are the containers of cards. They can also contain other nested decks. Various properties can be set on a deck to control how cards a reviewed, and how the cards are displayed on the deck page.
Decks fundamentally act as collections of cards. In addition to cards, decks can also contain other decks to create heirarchies of content. There are also a number of properties on the deck that control how cards are displayed and reviewd.
name
requiredparent-id
optionalsort
optionaltrashed?
optionalarchived?
optionalsort-by
optional:none :lexigraphically :lexicographically :created-at :updated-at :retention-rate-asc :interval-length
cards-view
optional:list :grid :note :column
.show-sides?
optionalsort-by-direction
optionalreview-reverse?
optionalExample request
POST http://localhost:8090/api/decks/
Content-Type: application/json
Accept: application/json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
{
"name": "My new deck",
"parent-id": "HJBMBGBO",
"sort": 1,
"archived?": true,
"trashed?": "2012-04-23T18:25:43.511Z",
"show-sides?": true,
"sort-by-direction": true,
"review-reverse?": true
}
POST http://localhost:8090/api/decks/
Content-Type: application/transit+json
Accept: application/transit+json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
{
"~:name": "My new deck",
"~:parent-id": "~:HJBMBGBO",
"~:sort": 1,
"~:archived?": true,
"~:sort-by": "~:lexicographically",
"~:trashed?": { "~#dt": 1645620058825 },
"~:cards-view": "~:list",
"~:show-sides?": true,
"~:sort-by-direction": true,
"~:review-reverse?": true
}
Returns the deck that was created, including its ID any any other fields that were populated automatically.
Example response
{
"sort-by-direction": true,
"name": "My new deck",
"show-sides?": true,
"cards-view": "list",
"id": "RMb5BZ67",
"review-reverse?": true,
"parent-id": "HJBMBGBO",
"sort-by": "lexicographically",
"archived?": true,
"sort": 1
}
{
"~:sort-by-direction": true,
"~:name": "My new deck",
"~:show-sides?": true,
"~:cards-view": "~:list",
"~:id": "~:6vXRDbq0",
"~:review-reverse?": true,
"~:parent-id": "~:HJBMBGBO",
"~:sort-by": "~:lexicographically",
"~:archived?": true,
"~:sort": 1
}
Retrieves a deck that already exists. The returned data will look identical to the data returned from the deck creation request.
No parameters.
Example request
GET https://app.mochi.cards/api/decks/btmZUXWM
Content-Type: application/json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
Returns the deck in whichever encoding is specified in the Accept
header.
Example response
{
"sort-by-direction": true,
"name": "My new deck",
"show-sides?": true,
"cards-view": "list",
"id": "RMb5BZ67",
"review-reverse?": true,
"parent-id": "HJBMBGBO",
"sort-by": "lexicographically",
"archived?": true,
"sort": 1
}
{
"~:sort-by-direction": true,
"~:name": "My new deck",
"~:show-sides?": true,
"~:cards-view": "~:list",
"~:id": "~:6vXRDbq0",
"~:review-reverse?": true,
"~:parent-id": "~:HJBMBGBO",
"~:sort-by": "~:lexicographically",
"~:archived?": true,
"~:sort": 1
}
:bookmark
optionalExample request
# Without a bookmark
GET https://app.mochi.cards/api/decks/
# With a bookmark
GET https://app.mochi.cards/api/decks?bookmark=somebookmarkgoeshere
A list of documents containing a bookmark.
Example response
{
"bookmark": "g1AAAABAeJzLYWBgYMpgSmHgKy5JLCrJTq2MT8lPzkzJBYpzGGY5BzimZYaD5Dlg8igyWQAw3xHv",
"docs": [
{
"id": "HJBMBGBO",
"sort": 6,
"name": "New Deck"
},
...
]
}
{
"~:bookmark": "g1AAAABAeJzLYWBgYMpgSmHgKy5JLCrJTq2MT8lPzkzJBYpzGGY5BzimZYaD5Dlg8igyWQAw3xHv",
"~:docs": [
{
"~:id": "~:HJBMBGBO",
"~:sort": 6,
"~:name": "New Deck"
},
...
...
]
}
name
optionalparent-id
optionalsort
optionaltrashed?
optionalarchived?
optionalsort-by
optional:none :lexigraphically :lexicographically :created-at :updated-at :retention-rate-asc :interval-length
cards-view
optional:list :grid :note :column
.show-sides?
optionalsort-by-direction
optionalreview-reverse?
optionalExample request
POST https://app.mochi.cards/api/decks/RMb5BZ67
Content-Type: application/json
Accept: application/json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
{
"name": "My new deck",
"parent-id": "HJBMBGBO",
"sort": 1,
"archived?": true,
"trashed?": "2012-04-23T18:25:43.511Z",
"show-sides?": true,
"sort-by-direction": true,
"review-reverse?": true
}
POST https://app.mochi.card/api/decks/
Content-Type: application/transit+json
Accept: application/transit+json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
{
"~:name": "My new deck",
"~:parent-id": "~:HJBMBGBO",
"~:sort": 1,
"~:archived?": true,
"~:sort-by": "~:lexicographically",
"~:trashed?": { "~#dt": 1645620058825 },
"~:cards-view": "~:list",
"~:show-sides?": true,
"~:sort-by-direction": true,
"~:review-reverse?": true
}
Returns the deck that was updated.
Example response
{
"sort-by-direction": true,
"name": "My new deck",
"show-sides?": true,
"cards-view": "list",
"id": "RMb5BZ67",
"review-reverse?": true,
"parent-id": "HJBMBGBO",
"sort-by": "lexicographically",
"archived?": true,
"sort": 1
}
{
"~:sort-by-direction": true,
"~:name": "My new deck",
"~:show-sides?": true,
"~:cards-view": "~:list",
"~:id": "~:6vXRDbq0",
"~:review-reverse?": true,
"~:parent-id": "~:HJBMBGBO",
"~:sort-by": "~:lexicographically",
"~:archived?": true,
"~:sort": 1
}
WARNING: This action cannot be undone. For a "soft" deletion you can update a deck and set its "trashed?" property to the current time.
None.
Example request
DELETE https://app.mochi.cards/api/decks/RMb5BZ67
Content-Type: application/json
Accept: application/json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
DELETE https://app.mochi.card/api/decks/
Content-Type: application/transit+json
Accept: application/transit+json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
Nothing.
Templates can be applied to decks and / or cards. Templates allow you to render content to cards using fields.
Retrieves a template that already exists.
No parameters.
Example request
GET https://app.mochi.cards/api/templates/btmZUXWM
Content-Type: application/json
Authorization: Basic NDZiYjA4YTY5ODAzNmFiMjk3MjU4ZWFlOg==
Returns the template in whichever encoding is specified in the Accept
header.
Example response
{
"id": "YDELNZSu",
"name": "Simple flashcard",
"content": "# << Front >>\n---\n<< Back >>",
"pos": "s",
"fields": {
"name": {
"id": "name",
"name": "Front",
"pos": "a"
},
"Ysrde7Lj": {
"id": "Ysrde7Lj",
"name": "Back",
"pos": "m",
"options": {
"multi-line?": true
}
}
}
}
{
"~:id": "~:YDELNZSu",
"~:name": "Simple flashcard",
"~:content": "# << Front >>\n---\n<< Back >>",
"~:pos": "s",
"~:fields": {
"~:name": {
"~:id": "~:name",
"~:name": "Front",
"~:pos": "a"
},
"~:Ysrde7Lj": {
"~:id": "~:Ysrde7Lj",
"~:name": "Back",
"~:pos": "m",
"~:options": {
"~:multi-line?": true
}
}
}
}
:bookmark
optionalExample request
# Without a bookmark
GET https://app.mochi.cards/api/templates/
# With a bookmark
GET https://app.mochi.cards/api/templates?bookmark=somebookmarkgoeshere
A list of documents containing a bookmark.
Example response
{
"bookmark": "g1AAAABAeJzLYWBgYMpgSmHgKy5JLCrJTq2MT8lPzkzJBYpzVBn4JgaaVZiC5Dlg8igyWQAxwRHd",
"docs": [
{
"id": "YDELNZSu",
"name": "Simple flashcard",
"content": "# << Front >>\n---\n<< Back >>",
"pos": "s",
"fields": {
"name": {
"id": "name",
"name": "Front",
"pos": "a"
},
"Ysrde7Lj": {
"id": "Ysrde7Lj",
"name": "Back",
"pos": "m",
"options": {
"multi-line?": true
}
}
}
},
...
]
}
{
"~:bookmark": "g1AAAABAeJzLYWBgYMpgSmHgKy5JLCrJTq2MT8lPzkzJBYpzVBn4JgaaVZiC5Dlg8igyWQAxwRHd",
"~:docs": [
{
"~:id": "~:YDELNZSu",
"~:name": "Simple flashcard",
"~:content": "# << Front >>\n---\n<< Back >>",
"~:pos": "s",
"~:fields": {
"~:name": {
"~:id": "~:name",
"~:name": "Front",
"~:pos": "a"
},
"~:Ysrde7Lj": {
"~:id": "~:Ysrde7Lj",
"~:name": "Back",
"~:pos": "m",
"~:options": {
"~:multi-line?": true
}
}
}
},
...
]
}