131 KiB
Most API methods are supported over the Websocket, HTTP, and MQTT
(if configured) transports. File Transfer and /access
requests are only
available over HTTP. The Websocket is required to receive server generated
events such as gcode responses. For information on how to set up the
Websocket, please see the Appendix at the end of this document.
HTTP API Overview
Moonraker's HTTP API could best be described as "RESTish". Attempts are made to conform to REST standards, however the dynamic nature of Moonraker's API registration along with the desire to keep consistency between mulitple API protocols results in an HTTP API that does not completely adhere to the standard.
Moonraker is capable of parsing request arguments from the both the body
(either JSON or form-data depending on the Content-Type
header) and from
the query string. All arguments are grouped together in one data structure,
with body arguments taking precedence over query arguments. Thus
if the same argument is supplied both in the body and in the
query string the body argument would be used. It is left up to the client
developer to decide exactly how they want to provide arguments, however
future API documention will make recommendations. As of March 1st 2021
this document exclusively illustrates arguments via the query string.
All successful HTTP requests will return a json encoded object in the form of:
{result: <response data>}
Response data is generally an object itself, however for some requests this may simply be an "ok" string.
Should a request result in an error, a standard error code along with an error specific message is returned.
Query string type hints
By default all arguments passed via the query string are represented as strings. Most endpoint handlers know the data type for each of their arguments, thus they can perform conversion from a string type if necessary. However some endpoints accept arguments of a "generic" type, thus the client is responsible for specifying the type if "string" is not desirable. This is not a problem for websocket requests as the JSON parser can extract the appropriate type. HTTP requests must provide "type hints" in these scenarios. Moonraker supplies support for the following query string type hints:
- int
- bool
- float
- json
The
json
type hint can be specified to pass an array or an object via the query string. Remember to percent encode the json string so that the query string is correctly parsed.
Type hints may be specified by post-fixing them to a key, with a ":"
separating the key and the hint. For example, lets assume that we
have a request that takes seconds
(integer) and enabled
(boolean)
arguments. The query string with type hints might look like:
?seconds:int=120&enabled:bool=true
A query string that takes a value
argument with which we want to
assing an object, {foo: 21.5, bar: "hello"}
might look like:
?value:json=%7B%22foo%22%3A21.5%2C%22bar%22%3A%22hello%22%7D
As you can see, a percent encoded json string is not human readable, thus using this functionality should be seen as a "last resort." If at all possible clients should attempt to put these arguments in the body of a request.
JSON-RPC API Overview
The Websocket and MQTT transports use the JSON-RPC 2.0
protocol. The Websocket transmits objects in a text frame, whereas MQTT
transmits them in the payload of a topic. When MQTT is configured Moonraker
subscribes to an api request topic. After an api request is processed Moonraker
publishes the return value to a response topic. By default these topics are
{instance_name}/moonraker/api/request
and
{instance_name}/moonraker/api/response
. The {instance_name}
should be a
unique identifier for each instance of Moonraker and defaults to the machine's
host name.
An encoded request should look something like:
{
"jsonrpc": "2.0",
"method": "API method",
"params": {"arg_one": 1, "arg_two": true},
"id": 354
}
The params
field may be left out if the API request takes no arguments.
The id
should be a unique value that has no chance of colliding
with other JSON-RPC requests. The method
is the API method, as defined
for each API in this document.
!!! tip
MQTT requests may provide an optional mqtt_timestamp
keyword
argument in the params
field of the JSON-RPC request. To avoid
potential collisions from time drift it is recommended to specify
the timestamp in microseconds since the Unix Epoch. If provided
Moonraker will use the timestamp to discard duplicate requests.
It is recommended to either provide a timestamp or publish API
requests at a QoS level of 0 or 2.
A successful request will return a response like the following:
{
"jsonrpc": "2.0",
"result": {"res_data": "success"},
"id": 354
}
The result
will generally contain an object, but as with the HTTP API in some
cases it may simply return a string. The id
field will return an id that
matches the one provided by the request.
Requests that result in an error will receive a properly formatted JSON-RPC response:
{
"jsonrpc": "2.0",
"error": {"code": 36000, "message": "Error Message"},
"id": 354
}
Some errors may not return a request ID, such as an improperly formatted request.
The moontest repo includes a basic test interface with example usage for most of the requests below. It also includes a basic JSON-RPC implementation that uses promises to return responses and errors (see json-rpc.js).
Jinja2 Template API Calls
Some template options in Moonraker's configuration, such as those in the
button component, may call Moonraker APIs through
the call_method(method_name, kwargs)
context function. The call_method
function takes the API's JSON-RPC method name as its first parameter, followed
by a set of keyword arguments as per the method's requirements.
# moonraker.conf
# Query Printer Objects example
[button check_status]
pin: gpio26
on_press:
{% set query_objs = {"toolhead": ["position"], "print_stats": None} %}
# JSON-RPC method is "printer.objects.query", which takes a single "objects"
# argument
{% set status = call_method("printer.objects.query", objects=query_objs) %}
# do something with the value returned from the object query, perhaps
# send a websocket notification or publish a mqtt topic
# Publish button event to MQTT Topic
[button check_status]
pin: gpio26
on_release:
# JSON-RPC method is "server.mqtt.publish"
{% do call_method("server.mqtt.publish",
topic="moonraker/mybutton",
payload="Button Released") %}
Printer Administration
Identify Connection
This method provides a way for persistent clients to identify themselves to Moonraker. This information may be used by Moonraker perform an action or present information based on if a specific client is connected. Currently this method is only available to websocket connections.
HTTP request: Not Available
JSON-RPC request (Websocket Only):
{
"jsonrpc": "2.0",
"method": "server.connection.identify",
"params": {
"client_name": "moontest",
"version": "0.0.1",
"type": "web",
"url": "http://github.com/arksine/moontest"
},
"id": 4656
}
All parameters are required. Below is an explanation of each parameter.
client_name
: The name of your client, such asMainsail
,Fluidd
,KlipperScreen
,MoonCord
, etc.version
: The current version of the connected clienttype
: Application type. May be one ofweb
,mobile
,desktop
,display
,bot
, orother
. These should be self explanatory, useother
if your client does not fit any of the prescribed options.url
: The url for your client's homepage
Returns:
The connection's unique identifer.
{
"connection_id": 1730367696
}
Get Websocket ID
!!! Warning This method is deprecated. Please use the identify endpoint to retreive the Websocket's UID
HTTP request: Not Available
JSON-RPC request (Websocket Only):
{
"jsonrpc": "2.0",
"method": "server.websocket.id",
"id": 4656
}
Returns:
The connected websocket's unique identifer.
{
"websocket_id": 1730367696
}
Get Klippy host information
HTTP Request:
GET /printer/info
JSON-RPC Request:
{
"jsonrpc": "2.0",
"method": "printer.info",
"id": 5445
}
Returns:
An object containing the build version, cpu info, Klippy's current state.
{
"state": "ready",
"state_message": "Printer is ready",
"hostname": "my-pi-hostname",
"software_version": "v0.9.1-302-g900c7396",
"cpu_info": "4 core ARMv7 Processor rev 4 (v7l)",
"klipper_path": "/home/pi/klipper",
"python_path": "/home/pi/klippy-env/bin/python",
"log_file": "/tmp/klippy.log",
"config_file": "/home/pi/printer.cfg",
}
Emergency Stop
HTTP request:
POST /printer/emergency_stop
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.emergency_stop",
"id": 4564
}
!!! note
This endpoint will immediately halt the printer and put it in a "shutdown"
state. It should be used to implement an "emergency stop" button and
also used if a user enters M112
(emergency stop) via a console.
Returns:
ok
Host Restart
HTTP request:
POST /printer/restart
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.restart",
"id": 4894
}
Returns:
ok
Firmware Restart
HTTP request:
POST /printer/firmware_restart
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.firmware_restart",
"id": 8463
}
Returns:
ok
Printer Status
List available printer objects
HTTP request:
GET /printer/objects/list
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.objects.list",
"id": 1454
}
Returns:
An array of "printer objects" that are currently available for query
or subscription. This list will be passed in an objects
parameter.
{
"objects": ["gcode", "toolhead", "bed_mesh", "configfile",...]
}
Query printer object status
HTTP request:
GET /printer/objects/query?gcode_move&toolhead&extruder=target,temperature
The above will request a status update for all gcode_move
and toolhead
attributes. Only the temperature
and target
attributes are requested
for the extruder
.
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.objects.query",
"params": {
"objects": {
"gcode_move": null,
"toolhead": ["position", "status"]
}
},
"id": 4654
}
!!! note
A null
value will fetch all available attributes for its key.
Returns:
An object where the top level items are "eventtime" and "status". The "status" item contains data about the requested update.
{
"eventtime": 578243.57824499,
"status": {
"gcode_move": {
"absolute_coordinates": true,
"absolute_extrude": true,
"extrude_factor": 1,
"gcode_position": [0, 0, 0, 0],
"homing_origin": [0, 0, 0, 0],
"position": [0, 0, 0, 0],
"speed": 1500,
"speed_factor": 1,
},
"toolhead": {
"position": [0, 0, 0, 0],
"status": "Ready"
}
}
}
See printer_objects.md for details on the printer objects available for query.
Subscribe to printer object status
HTTP request:
POST /printer/objects/subscribe?connection_id=123456789&gcode_move&extruder`
!!! note
The HTTP API requires that a connection_id
is passed via the query
string or as part of the form. This should be the
ID reported from a currently connected websocket. A
request that includes only the connection_id
argument will cancel the
subscription on the specified websocket.
This request is not available over MQTT as it can not be set per client.
Instead MQTT can publish printer status by setting the `status_objects`
option in the `[mqtt]` section.
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.objects.subscribe",
"params": {
"objects": {
"gcode_move": null,
"toolhead": ["position", "status"]
}
},
"id": 5434
}
!!! note
If objects
is set to an empty object then the subscription will
be cancelled.
Returns:
Status data for objects in the request, with the format matching that of
the /printer/objects/query
:
{
"eventtime": 578243.57824499,
"status": {
"gcode_move": {
"absolute_coordinates": true,
"absolute_extrude": true,
"extrude_factor": 1,
"gcode_position": [0, 0, 0, 0],
"homing_origin": [0, 0, 0, 0],
"position": [0, 0, 0, 0],
"speed": 1500,
"speed_factor": 1,
},
"toolhead": {
"position": [0, 0, 0, 0],
"status": "Ready"
}
}
}
See printer_objects.md for details on the printer objects available for subscription.
Status updates for subscribed objects are sent asynchronously over the websocket. See the notify_status_update notification for details.
Query Endstops
HTTP request:
GET /printer/query_endstops/status
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.query_endstops.status",
"id": 3456
}
Returns:
An object containing the current endstop state, where each field is an endstop identifier, with a string value of "open" or "TRIGGERED".
{
"x": "TRIGGERED",
"y": "open",
"z": "open"
}
Query Server Info
HTTP request:
GET /server/info
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.info",
"id": 9546
}
Returns:
An object containing various fields that report server state.
{
"klippy_connected": true,
"klippy_state": "ready",
"components": [
"database",
"file_manager",
"klippy_apis",
"machine",
"data_store",
"shell_command",
"proc_stats",
"history",
"octoprint_compat",
"update_manager",
"power"
],
"failed_components": [],
"registered_directories": ["config", "gcodes", "config_examples", "docs"],
"warnings": [
"Invalid config option 'api_key_path' detected in section [authorization]. Remove the option to resolve this issue. In the future this will result in a startup error.",
"Unparsed config section [fake_section] detected. This may be the result of a component that failed to load. In the future this will result in a startup error."
],
"websocket_count": 2,
"moonraker_version": "v0.7.1-105-ge4f103c",
"api_version": [1, 0, 0],
"api_version_string": "1.0.0"
}
!!! warning
This object also includes plugins
and failed_plugins
fields that
are now deprecated. They duplicate the information in
components
and failed_components
, and will be removed in the future.
Note that klippy_state
will match the state
value received from
/printer/info
. The klippy_connected
item tracks the state of the
unix domain socket connect to Klippy. The components
key will return a list
of enabled components. This can be used by clients to check if an optional
component is available. Optional components that do not load correctly will
not prevent the server from starting, thus any components that failed to load
will be reported in the failed_components
field.
The websocket_count
field reports the total number of connected websockets.
Get Server Configuration
HTTP request:
GET /server/config
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.config",
"id": 5616,
}
Returns:
An object containing the server's full configuration. Note that
this includes auxiliary configuration sections not part of moonraker.conf
,
for example the update_manager static debian moonraker
section.
Options not specified in moonraker.conf
with default values are also
included.
{
"config": {
"server": {
"host": "0.0.0.0",
"port": 7125,
"klippy_uds_address": "/tmp/klippy_uds",
"max_upload_size": 210,
"enable_debug_logging": true,
"database_path": "~/.moonraker_database",
"config_path": "~/printer_config",
"temperature_store_size": 100,
"gcode_store_size": 50
},
"authorization": {
"api_key_file": "~/.moonraker_api_key",
"enabled": true,
"cors_domains": "\nhttp://my.mainsail.xyz\nhttp://app.fluidd.xyz",
"trusted_clients": "\n192.168.1.0/24"
},
"system_args": {},
"history": {},
"octoprint_compat": {},
"update_manager": {
"enable_auto_refresh": true,
"distro": "debian",
"enable_repo_debug": true,
"client_repo": null
},
"update_manager static debian moonraker": {},
"update_manager client mainsail": {
"type": "web",
"repo": "mainsail-crew/mainsail",
"path": "~/mainsail",
"persistent_files": null
},
"update_manager client fluidd": {
"type": "web",
"repo": "fluidd-core/fluidd",
"path": "~/fluidd",
"persistent_files": null
},
"power green_led": {
"type": "gpio",
"locked_while_printing": false,
"off_when_shutdown": false,
"restart_klipper_when_powered": false,
"pin": "gpiochip0/gpio26",
"initial_state": false
},
"update_manager static debian klipper": {}
}
}
Request Cached Temperature Data
HTTP request:
GET /server/temperature_store
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.temperature_store",
"id": 2313
}
Returns:
An object where the keys are the available temperature sensor names, and with the value being an array of stored temperatures. The array is updated every 1 second by default, containing a total of 1200 values (20 minutes). The array is organized from oldest temperature to most recent (left to right). Note that when the host starts each array is initialized to 0s.
{
"extruder": {
"temperatures": [21.05, 21.12, 21.1, 21.1, 21.1],
"targets": [0, 0, 0, 0, 0],
"powers": [0, 0, 0, 0, 0]
},
"temperature_fan my_fan": {
"temperatures": [21.05, 21.12, 21.1, 21.1, 21.1],
"targets": [0, 0, 0, 0, 0],
"speeds": [0, 0, 0, 0, 0],
},
"temperature_sensor my_sensor": {
"temperatures": [21.05, 21.12, 21.1, 21.1, 21.1]
}
}
Request Cached GCode Responses
HTTP request:
GET /server/gcode_store?count=100
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.gcode_store",
"params": {
"count": 100
},
"id": 7643}
The count
argument is optional, limiting number of returned items
in the response to the value specified. If omitted, the entire gcode
store will be returned (up to 1000 responses).
Returns:
An object with the field gcode_store
that contains an array
of objects. Each object will contain message
, time
, and
type
fields. The time
field is reported in Unix Time.
The type
field will either be command
or response
.
{
"gcode_store": [
{
"message": "FIRMWARE_RESTART",
"time": 1615832299.1167388,
"type": "command"
},
{
"message": "// Klipper state: Ready",
"time": 1615832309.9977088,
"type": "response"
},
{
"message": "M117 This is a test",
"time": 1615834094.8662775,
"type": "command"
},
{
"message": "G4 P1000",
"time": 1615834098.761729,
"type": "command"
},
{
"message": "STATUS",
"time": 1615834104.2860553,
"type": "command"
},
{
"message": "// Klipper state: Ready",
"time": 1615834104.3299904,
"type": "response"
}
]
}
Restart Server
HTTP request:
POST /server/restart
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.restart",
"id": 4656
}
Returns:
ok
upon receipt of the restart request. After the request
is returns, the server will restart. Any existing connection
will be disconnected. A restart will result in the creation
of a new server instance where the configuration is reloaded.
GCode APIs
Run a gcode:
HTTP request:
POST /printer/gcode/script?script=G28
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.gcode.script",
"params": {
"script": "G28"
},
"id": 7466}
!!! warning
When M112
(emergency stop) is requested via this endpoint it will not
immediately stop the printer. M112
will be placed on the gcode queue and
executed after all previous gcodes are complete. If a client detects
M112
via user input (such as a console) it should request the
/printer/emergency_stop
endpoint to immediately halt the printer. This
may be done in addition to sending the M112
gcode if desired.
Returns:
ok
when the gcode has completed execution.
Get GCode Help
HTTP request:
GET /printer/gcode/help
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.gcode.help",
"id": 4645
}
Returns:
An object where they keys are gcode handlers and values are the associated help strings. Note that help strings are not available for default gcode handlers such as G1, G28, etc, nor are they available for extended handlers that failed to register a description in Klippy.
{
"RESTORE_GCODE_STATE": "Restore a previously saved G-Code state",
"PID_CALIBRATE": "Run PID calibration test",
"QUERY_ADC": "Report the last value of an analog pin",
"TUNING_TOWER": "Tool to adjust a parameter at each Z height",
"SAVE_CONFIG": "Overwrite config file and restart",
"SET_DISPLAY_GROUP": "Set the active display group",
"SAVE_GCODE_STATE": "Save G-Code coordinate state",
"SET_PRESSURE_ADVANCE": "Set pressure advance parameters",
"SET_GCODE_OFFSET": "Set a virtual offset to g-code positions",
"BED_TILT_CALIBRATE": "Bed tilt calibration script",
...
}
Print Management
Print a file
HTTP request:
POST /printer/print/start?filename=test_print.gcode
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.print.start",
"params": {
"filename": "test_pring.gcode"
},
"id": 4654
}
Returns:
ok
Pause a print
HTTP request:
POST /printer/print/pause
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.print.pause",
"id": 4564
}
Returns:
ok
Resume a print
HTTP request:
POST /printer/print/resume
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.print.resume",
"id": 1465
}
Returns:
ok
Cancel a print
HTTP request:
POST /printer/print/cancel
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "printer.print.cancel",
"id": 2578
}
Returns:
ok
Machine Commands
Get System Info
HTTP request:
GET /machine/system_info
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.system_info",
"id": 4665
}
Returns: Information about the host system in the following format:
{
"system_info": {
"cpu_info": {
"cpu_count": 4,
"bits": "32bit",
"processor": "armv7l",
"cpu_desc": "ARMv7 Processor rev 4 (v7l)",
"serial_number": "b898bdb4",
"hardware_desc": "BCM2835",
"model": "Raspberry Pi 3 Model B Rev 1.2",
"total_memory": 945364,
"memory_units": "kB"
},
"sd_info": {
"manufacturer_id": "03",
"manufacturer": "Sandisk",
"oem_id": "5344",
"product_name": "SU32G",
"product_revision": "8.0",
"serial_number": "46ba46",
"manufacturer_date": "4/2018",
"capacity": "29.7 GiB",
"total_bytes": 31914983424
},
"distribution": {
"name": "Raspbian GNU/Linux 10 (buster)",
"id": "raspbian",
"version": "10",
"version_parts": {
"major": "10",
"minor": "",
"build_number": ""
},
"like": "debian",
"codename": "buster"
},
"available_services": [
"klipper",
"klipper_mcu",
"moonraker"
],
"service_state": {
"klipper": {
"active_state": "active",
"sub_state": "running"
},
"klipper_mcu": {
"active_state": "active",
"sub_state": "running"
},
"moonraker": {
"active_state": "active",
"sub_state": "running"
}
},
"virtualization": {
"virt_type": "none",
"virt_identifier": "none"
},
"python": {
"version": [
3,
7,
3,
"final",
0
],
"version_string": "3.7.3 (default, Jan 22 2021, 20:04:44) [GCC 8.3.0]"
},
"network": {
"wlan0": {
"mac_address": "<redacted_mac>",
"ip_addresses": [
{
"family": "ipv4",
"address": "192.168.1.127",
"is_link_local": false
},
{
"family": "ipv6",
"address": "<redacted_ipv6>",
"is_link_local": false
},
{
"family": "ipv6",
"address": "fe80::<redacted>",
"is_link_local": true
}
]
}
}
}
}
!!! note
If no SD Card is detected the sd_info
field will contain an empty object.
Shutdown the Operating System
HTTP request:
POST /machine/shutdown
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.shutdown",
"id": 4665
}
Returns:
This request will not return. The machine will shutdown and the socket connection will drop.
Reboot the Operating System
HTTP request:
POST /machine/reboot
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.reboot",
"id": 4665
}
Returns:
This request will not return. The machine will reboot and the socket connection will drop.
Restart a system service
Restarts a system service via sudo systemctl restart {name}
. Currently
the moonraker
, klipper
, MoonCord
, KlipperScreen
and webcamd
services are supported.
HTTP request:
POST /machine/services/restart?service={name}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.services.restart",
"params": {
"service": "{name}"
},
"id": 4656}
Returns:
ok
when complete. Note that if moonraker
is chosen, the return
value will be sent prior to the service restart.
Stop a system service
Stops a system service via sudo systemctl stop <name>
. Currently
only webcamd
and klipper
are supported.
HTTP request:
POST /machine/services/stop?service={name}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.services.stop",
"params": {
"service": "{name}"
},
"id": 4645
}
Returns:
ok
Start a system service
Starts a system service via sudo systemctl start <name>
. Currently
only webcamd
and klipper
are supported.
HTTP request:
POST /machine/services/start?service={name}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.services.start",
"params": {
"service": "{name}"
},
"id": 4645
}
Returns:
ok
Get Moonraker Process Stats
Returns system usage information about the moonraker process.
HTTP request:
GET /machine/proc_stats
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.proc_stats",
"id": 7896
}
Returns:
An object in the following format:
{
"moonraker_stats": [
{
"time": 1626612666.850755,
"cpu_usage": 2.66,
"memory": 24732,
"mem_units": "kB"
},
{
"time": 1626612667.8521338,
"cpu_usage": 2.62,
"memory": 24732,
"mem_units": "kB"
}
],
"throttled_state": {
"bits": 0,
"flags": []
},
"cpu_temp": 45.622,
"network": {
"lo": {
"rx_bytes": 113516429,
"tx_bytes": 113516429,
"bandwidth": 3342.68
},
"wlan0": {
"rx_bytes": 48471767,
"tx_bytes": 113430843,
"bandwidth": 4455.91
}
},
"system_cpu_usage": {
"cpu": 2.53,
"cpu0": 3.03,
"cpu1": 5.1,
"cpu2": 1.02,
"cpu3": 1
},
"system_uptime": 2876970.38089603,
"websocket_connections": 4
}
Process information is sampled every second. The moonraker_stats
field
will return up to 30 samples, each sample with the following fields:
time
: Time of the sample (in seconds since the Epoch)cpu_usage
: A floating point value between 0-100, representing the CPU usage of the Moonraker process.memory
: Integer value representing the current amount of memory allocated in RAM (resident set size).mem_units
: A string indentifying the units of the value in thememory
field. This is typically "kB", but not guaranteed.
If the system running Moonraker supports vcgencmd
then Moonraker
will check the current throttled flags via vcgencmd get_throttled
and report them in the throttled_state
field:
bits
: An integer value that represents the bits reported byvcgencmd get_throttled
flags
: Descriptive flags parsed out of the bits. One or more of the following flags may be reported:- "Under-Voltage Detected"
- "Frequency Capped"
- "Currently Throttled"
- "Temperature Limit Active"
- "Previously Under-Volted"
- "Previously Frequency Capped"
- "Previously Throttled"
- "Previously Temperature Limited"
The first four flags indicate an active throttling condition,
whereas the last four indicate a previous condition (may or
may not still be active). If vcgencmd
is not available
throttled_state
will report null
.
If the system reports CPU temp at /sys/class/thermal/thermal_zone0
then temperature will be supplied in the cpu_temp
field. Otherwise
the field will be set to null
.
If the system reports network statistics at /proc/net/dev
then the
network
field will contain network statistics. All available interfaces
will be tracked. Each interface reports the following fields:
rx_bytes
: total number of bytes received over the interfacetx_bytes
: total number of bytes transferred over the interfacebandwidth
: estimated current bandwidth used (both rx and tx) in bytes/second
If network information is not available then the network
field will
contain an empty object.
If the system reports cpu usage at /proc/stat
then the system_cpu_usage
field will contain an object with cpu usage data. The cpu
field of this
object reports total cpu usage, while each cpuX
field is usage per core.
The websocket_connections
field reports the number of active websockets
currently connected to moonraker.
File Operations
Most file operations are available over both APIs, however file upload and file download are currently only available via HTTP APIs.
Moonraker organizes local directories into "roots". For example,
gcodes are located at http:\\host\server\files\gcodes\*
, otherwise known
as the "gcodes" root. The following roots are available:
- gcodes
- config
- config_examples (read-only)
- docs (read-only)
Write operations (upload, delete, make directory, remove directory) are
only available on the gcodes
and config
roots. Note that the config
root
is only available if the config_path
option has been set in Moonraker's
configuration.
List available files
Walks through a directory and fetches all files. All file names include a
path relative to the specified root
.
HTTP request:
GET /server/files/list?root={root_folder}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.files.list",
"params": {
"root": "{root_folder}"
},
"id": 4644
}
!!! tip
If the root
argument is omitted the request will default to
the gcodes
root.
!!! note
The gcodes
root will only return files with valid gcode
extensions.
Returns: A list of objects, where each object contains file data.
[
{
"path": "3DBenchy_0.15mm_PLA_MK3S_2h6m.gcode",
"modified": 1615077020.2025201,
"size": 4926481,
"permissions": "rw"
},
{
"path": "Shape-Box_0.2mm_PLA_Ender2_20m.gcode",
"modified": 1614910966.946807,
"size": 324236,
"permissions": "rw"
},
{
"path": "test_dir/A-Wing.gcode",
"modified": 1605202259,
"size": 1687387,
"permissions": "rw"
},
{
"path": "test_dir/CE2_CubeTest.gcode",
"modified": 1614644445.4025,
"size": 1467339,
"permissions": "rw"
},
{
"path": "test_dir/V350_Engine_Block_-_2_-_Scaled.gcode",
"modified": 1615768477.5133543,
"size": 189713016,
"permissions": "rw"
},
]
Get gcode metadata
Get metadata for a specified gcode file. If the file is located in a subdirectory, then the file name should include the path relative to the "gcodes" root. For example, if the file is located at:
http://host.local/server/files/gcodes/my_sub_dir/my_print.gcode
Then the {filename}
argument should be my_sub_dir/my_print.gcode
.
HTTP request:
GET /server/files/metadata?filename={filename}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.files.metadata",
"params": {
"filename": "{filename}"
},
"id": 3545
}
Returns:
Metadata for the requested file if it exists. If any fields failed parsing they will be omitted. The metadata will always include the file name, modified time, and size.
{
"print_start_time": null,
"job_id": null,
"size": 4926481,
"modified": 1615077020.2025201,
"slicer": "SuperSlicer",
"slicer_version": "2.2.52",
"layer_height": 0.15,
"first_layer_height": 0.2,
"object_height": 48.05,
"filament_total": 4056.4,
"estimated_time": 7569,
"thumbnails": [
{
"width": 32,
"height": 32,
"size": 2596,
"relative_path": ".thumbs/3DBenchy_0.15mm_PLA_MK3S_2h6m-32x32.png"
},
{
"width": 400,
"height": 300,
"size": 73308,
"relative_path": ".thumbs/3DBenchy_0.15mm_PLA_MK3S_2h6m-400x300.png"
}
],
"first_layer_bed_temp": 60,
"first_layer_extr_temp": 215,
"gcode_start_byte": 79451,
"gcode_end_byte": 4915668,
"filename": "3DBenchy_0.15mm_PLA_MK3S_2h6m.gcode"
}
!!! note
The print_start_time
and job_id
fields are initialized to
null
. They will be updated for each print job if the user has the
[history]
component configured
Get directory information
Returns a list of files and subdirectories given a supplied path.
Unlike /server/files/list
, this command does not walk through
subdirectories. This request will return all files in a directory,
including files in the gcodes
root that do not have a valid gcode
extension.
HTTP request:
GET /server/files/directory?path=gcodes/my_subdir&extended=true
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.files.get_directory",
"params": {
"path": "gcodes/my_subdir",
"extended": true
},
"id": 5644
}
!!! tip
If the path
argument is omitted then the command will return
directory information from the gcodes
root.
The extended
argument is optional and defaults to false. If
supplied and set to true then data returned for gcode files
will also include metadata (if available).
Returns:
An object containing file and subdirectory information in the following format:
{
"dirs": [
{
"modified": 1615768162.0412788,
"size": 4096,
"permissions": "rw",
"dirname": "test"
},
{
"modified": 1613569827.489749,
"size": 4096,
"permissions": "rw",
"dirname": "Cura"
},
{
"modified": 1615767459.6265886,
"size": 4096,
"permissions": "rw",
"dirname": "thumbs"
}
],
"files": [
{
"modified": 1615578004.9639666,
"size": 7300692,
"permissions": "rw",
"filename": "Funnel_0.2mm_PLA_Ender2_2h4m.gcode"
},
{
"modified": 1589156863.9726968,
"size": 4214831,
"permissions": "rw",
"filename": "CE2_Pi3_A+_CaseLID.gcode"
},
{
"modified": 1615030592.7722695,
"size": 2388774,
"permissions": "rw",
"filename": "CE2_calicat.gcode"
},
],
"disk_usage": {
"total": 7522213888,
"used": 4280369152,
"free": 2903625728
},
"root_info": {
"name": "gcodes",
"permissions": "rw"
}
}
Create directory
Creates a directory at the specified path.
HTTP request:
POST /server/files/directory?path=gcodes/my_new_dir
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.files.post_directory",
"params": {
"path": "gcodes/my_new_dir"
},
"id": 6548
}
Returns: Information about the created directory
{
"item": {
"path": "gcodes/testdir",
"root": "gcodes"
},
"action": "create_dir"
}
Delete directory
Deletes a directory at the specified path.
HTTP request:
DELETE /server/files/directory?path=gcodes/my_subdir&force=false
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.files.delete_directory",
"params": {
"path": "gcodes/my_new_dir",
"force": false
},
"id": 6545
}
!!! warning
If the specified directory contains files then the delete request
will fail unless the force
argument is set to true
.
Returns: Information about the deleted directory
{
"item": {
"path": "gcodes/testdir",
"root": "gcodes"
},
"action": "delete_dir"
}
Move a file or directory
Moves a file or directory from one location to another. The following conditions must be met for a move successful move:
- The source must exist
- The user (typically "pi") must have the appropriate file permissions
- Neither the source nor destination can be loaded by the
virtual_sdcard
. If the source is a directory, it must not contain a file loaded by thevirtual_sdcard
.
When specifying the source
and dest
, the root
directory should be
prefixed. Currently the only supported roots for dest
are gcodes
"
and config
".
This API may also be used to rename a file or directory. Be aware that an attempt to rename a directory to a directory that already exists will result in moving the source directory into the destination directory.
HTTP request:
POST /server/files/move?source=gcodes/my_file.gcode&dest=gcodes/subdir/my_file.gcode
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.files.move",
"params": {
"source": "gcodes/my_file.gcode",
"dest": "gcodes/subdir/my_file.gcode"
},
"id": 5664
}
Returns: Information about the moved file or directory
{
"result": {
"item": {
"root": "gcodes",
"path": "test4/test3"
},
"source_item": {
"path": "gcodes/test4/test3",
"root": "gcodes"
},
"action": "move_dir"
}
}
Copy a file or directory
Copies a file or directory from one location to another. A successful copy has
the pre-requesites as a move with one exception, a copy may complete if the
source file or directory is loaded by the virtual_sdcard
. As with the move
API, the source
and dest
should have the root prefixed to the path.
HTTP request:
POST /server/files/copy?source=gcodes/my_file.gcode&dest=gcodes/subdir/my_file.gcode
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.files.copy",
"params": {
"source": "gcodes/my_file.gcode",
"dest": "gcodes/subdir/my_file.gcode"
},
"id": 5623
}
Returns: Information about the copied file or directory
{
"item": {
"root": "gcodes",
"path": "test4/Voron_v2_350_aferburner_Filament Cover_0.2mm_ABS.gcode"
},
"action": "create_file"
}
File download
Retrieves file filename
at root root
. The filename
must include
the relative path if it is not in the root folder.
HTTP request:
GET /server/files/{root}/{filename}
JSON-RPC request: Not Available
Returns:
The requested file
File upload
Upload a file. Currently files may be uploaded to the gcodes
or config
roots, with gcodes
being the default. If one wishes to upload
to a subdirectory, the path may be added to the upload's file name
(relative to the root). If the directory does not exist an error will be
returned. Alternatively, the path
form argument may be set, as explained
below.
HTTP request:
POST /server/files/upload`
Content-Type: multipart/form-data
------FormBoundaryemap3PkuvKX0B3HH
Content-Disposition: form-data; name="file"; filename="myfile.gcode"
Content-Type: application/octet-stream
<binary data>
------FormBoundaryemap3PkuvKX0B3HH--
The file must be uploaded in the request's body multipart/form-data
(ie:
<input type="file">
). The following arguments may also be added to the
form-data:
root
: The root location in which to upload the file. Currently this may begcodes
orconfig
. If not specified the default isgcodes
.path
: This argument may contain a path (relative to the root) indicating a subdirectory to which the file is written. If apath
is present the server will attempt to create any subdirectories that do not exist.checksum
: A SHA256 hex digest calculated by the client for the uploaded file. If this argument is supplied the server will compare it to its own checksum calculation after the upload has completed. A checksum mismatch will result in a 422 error.
Arguments available only for the gcodes
root:
print
: If set to "true", Klippy will attempt to start the print after uploading. Note that this value should be a string type, not boolean. This provides compatibility with OctoPrint's upload API.
JSON-RPC request: Not Available
Returns: Information about the uploaded file. Note that print_started
is only included when the supplied root is set to gcodes
.
{
"item": {
"path": "Lock Body Shim 1mm_0.2mm_FLEX_MK3S_2h30m.gcode",
"root": "gcodes"
},
"print_started": false,
"action": "create_file"
}
File delete
Delete a file in the requested root. If the file exists in a subdirectory,
its relative path must be part of the {filename}
argument.
HTTP request:
DELETE /server/files/{root}/{filename}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.files.delete_file",
"params": {
"path": "{root}/{filename}"
},
"id": 1323
}
Returns: Information about the deleted file
{
"item": {
"path": "Lock Body Shim 1mm_0.2mm_FLEX_MK3S_2h30m.gcode",
"root": "gcodes"
},
"action": "delete_file"
}
Download klippy.log
HTTP request:
GET /server/files/klippy.log
JSON-RPC request: Not Available
Returns:
The requested file
Download moonraker.log
HTTP request:
GET /server/files/moonraker.log
JSON-RPC request: Not Available
Returns:
The requested file
Authorization
The Authorization endpoints are enabled when the user has the
[authorization]
component configured in moonraker.conf
.
Untrusted clients must use either a JSON Web Token or an API key to access
Moonraker's HTTP APIs. JWTs should be included in the Authorization
header as a Bearer
type for each HTTP request. If using an API Key it
should be included in the X-Api-Key
header for each HTTP Request.
!!! note
For requests in which clients cannot modify headers it is acceptable
to pass the JWT via the query string's access_token
argument.
Alternatively client developers may request a oneshot_token
and
send the result via the token
query string argument.
!!! warning It is strongly recommended that arguments for the below APIs are passed in the request's body.
Login User
HTTP Request:
POST /access/login
Content-Type: application/json
{
"username": "my_user",
"password": "my_password"
}
JSON-RPC request: Not Available
Returns: An object the logged in username, auth token, refresh token, and action summary:
{
"username": "my_user",
"token": "eyJhbGciOiAiSFMyNTYiLCAidHlwIjogIkpXVCJ9.eyJpc3MiOiAiTW9vbnJha2VyIiwgImlhdCI6IDE2MTg4NzY4MDAuNDgxNjU1LCAiZXhwIjogMTYxODg4MDQwMC40ODE2NTUsICJ1c2VybmFtZSI6ICJteV91c2VyIiwgInRva2VuX3R5cGUiOiAiYXV0aCJ9.QdieeEskrU0FrH7rXKuPDSZxscM54kV_vH60uJqdU9g",
"refresh_token": "eyJhbGciOiAiSFMyNTYiLCAidHlwIjogIkpXVCJ9.eyJpc3MiOiAiTW9vbnJha2VyIiwgImlhdCI6IDE2MTg4NzY4MDAuNDgxNzUxNCwgImV4cCI6IDE2MjY2NTI4MDAuNDgxNzUxNCwgInVzZXJuYW1lIjogIm15X3VzZXIiLCAidG9rZW5fdHlwZSI6ICJyZWZyZXNoIn0.btJF0LJfymInhGJQ2xvPwkp2dFUqwgcw4OA_wE-EcCM",
"action": "user_logged_in"
}
- The
token
field is a JSON Web Token used to authorize the user. It should be included in theAuthorization
header as aBearer
type for all HTTP requests. Thetoken
expires after 1 hour. - The
refresh_token
field contains a JWT that can be used to generate new tokens after they are expire. See the refresh token section for details.
!!! Note This endpoint may be accessed by unauthorized clients. A 401 would only be returned if the username and/or password is invalid.
Logout Current User
HTTP Request:
POST /access/logout
JSON-RPC request: Not Available
Returns: An object containing the logged out username and action summary.
{
"username": "my_user",
"action": "user_logged_out"
}
Get Current User
HTTP Request:
GET /access/user
JSON-RPC request: Not Available
Returns: An object containing the currently logged in user name and the date on which the user was created (in unix time).
{
"username": "my_user",
"created_on": 1618876783.8896716
}
Create User
HTTP Request:
POST /access/user
Content-Type: application/json
{
"username": "my_user",
"password": "my_password"
}
JSON-RPC request: Not Available
Returns: An object containing the created user name, an auth token, a refresh token, and an action summary. Creating a user also effectively logs the user in.
{
"username": "my_user",
"token": "eyJhbGciOiAiSFMyNTYiLCAidHlwIjogIkpXVCJ9.eyJpc3MiOiAiTW9vbnJha2VyIiwgImlhdCI6IDE2MTg4NzY3ODMuODkxNjE5LCAiZXhwIjogMTYxODg4MDM4My44OTE2MTksICJ1c2VybmFtZSI6ICJteV91c2VyIiwgInRva2VuX3R5cGUiOiAiYXV0aCJ9.oH0IShTL7mdlVs4kcx3BIs_-1j0Oe-qXezJKjo-9Xgo",
"refresh_token": "eyJhbGciOiAiSFMyNTYiLCAidHlwIjogIkpXVCJ9.eyJpc3MiOiAiTW9vbnJha2VyIiwgImlhdCI6IDE2MTg4NzY3ODMuODkxNzAyNCwgImV4cCI6IDE2MjY2NTI3ODMuODkxNzAyNCwgInVzZXJuYW1lIjogIm15X3VzZXIiLCAidG9rZW5fdHlwZSI6ICJyZWZyZXNoIn0.a6ZeRjk8RQQJDDH0JV-qGY_d_HIgfI3XpsqUlUaFT7c",
"action": "user_created"
}
!!! note
Unlike /access/login
, /access/user
is a protected endpoint. To
create a new user a client must either be trusted, use the API Key,
or be logged in as another user.
Delete User
Deletes a registered user.
!!! note A request to delete a user MUST come from an authorized source other than the account to be deleted. This can be a "trusted user", the "api key user", or any other user account.
HTTP Request:
DELETE /access/user
Content-Type: application/json
{
"username": "my_username"
}
JSON-RPC request: Not Available
Returns: The username of the deleted user and an action summary. This effectively logs the user out, as all outstanding tokens will be invalid.
{
"username": "my_user",
"action": "user_deleted"
}
List Available Users
HTTP Request:
GET /access/users/list
JSON-RPC request: Not Available
Returns: A list of created users on the system
{
"users": [
{
"username": "testuser",
"created_on": 1618771331.1685035
},
{
"username": "testuser2",
"created_on": 1620943153.0191233
}
]
}
Reset User Password
HTTP Request:
POST /access/user/password
Content-Type: application/json
{
"password": "my_current_password",
"new_password": "my_new_pass"
}
JSON-RPC request: Not Available
Returns: The username and action summary.
{
"username": "my_user",
"action": "user_password_reset"
}
Refresh JSON Web Token
This endpoint can be used to refresh an expired auth token. If this request returns an error then the refresh token is no longer valid and the user must login with their credentials.
HTTP Request:
POST /access/refresh_jwt
Content-Type: application/json
{
"refresh_token": "eyJhbGciOiAiSFMyNTYiLCAidHlwIjogIkpXVCJ9.eyJpc3MiOiAiTW9vbnJha2VyIiwgImlhdCI6IDE2MTg4Nzc0ODUuNzcyMjg5OCwgImV4cCI6IDE2MjY2NTM0ODUuNzcyMjg5OCwgInVzZXJuYW1lIjogInRlc3R1c2VyIiwgInRva2VuX3R5cGUiOiAicmVmcmVzaCJ9.Y5YxGuYSzwJN2WlunxlR7XNa2Y3GWK-2kt-MzHvLbP8"
}
JSON-RPC request: Not Available
Returns: The username, new auth token, and action summary.
{
"username": "my_user",
"token": "eyJhbGciOiAiSFMyNTYiLCAidHlwIjogIkpXVCJ9.eyJpc3MiOiAiTW9vbnJha2VyIiwgImlhdCI6IDE2MTg4NzgyNDMuNTE2Nzc5MiwgImV4cCI6IDE2MTg4ODE4NDMuNTE2Nzc5MiwgInVzZXJuYW1lIjogInRlc3R1c2VyIiwgInRva2VuX3R5cGUiOiAiYXV0aCJ9.Ia_X_pf20RR4RAEXcxalZIOzOBOs2OwearWHfRnTSGU",
"action": "user_jwt_refresh"
}
!!! Note This endpoint may be accessed by unauthorized clients. A 401 would only be returned if the refresh token is invalid.
Generate a Oneshot Token
Javascript is not capable of modifying the headers for some HTTP requests
(for example, the websocket
), which is a requirement to apply JWT or API Key
authorization. To work around this clients may request a Oneshot Token and
pass it via the query string for these requests. Tokens expire in 5 seconds
and may only be used once, making them relatively safe for inclusion in the
query string.
HTTP request:
GET /access/oneshot_token
JSON-RPC request: Not Available
Returns:
A temporary token that may be added to a request's query string for access to any API endpoint. The query string should be added in the form of:
?token={base32_ramdom_token}
Get the Current API Key
HTTP request:
GET /access/api_key
JSON-RPC request: Not Available
Returns:
The current API key
Generate a New API Key
HTTP request:
POST /access/api_key
JSON-RPC request: Not Available
Returns:
The newly generated API key. This overwrites the previous key. Note that the API key change is applied immediately, all subsequent HTTP requests from untrusted clients must use the new key.
Database APIs
The following endpoints provide access to Moonraker's ldbm database. The
database is divided into namespaces
. Each client may define its own
namespace to store information. From the client's point of view, a
namespace is an object
. Items in the database are accessed by providing
a namespace and a key. A key may be specifed as string, where a "." is a
delimeter, to access nested fields. Alternatively the key may be specified
as an array of strings, where each string references a nested field.
This is useful for scenarios where your namespace contains keys that include
a "." character.
!!! note
Moonraker reserves the moonraker
, gcode_metadata
, and history
namespaces. Clients may read from these namespaces but they may not
modify them.
For example, assume the following object is stored in the "superclient" namespace:
{
"settings": {
"console": {
"enable_autocomplete": true
}
},
"theme": {
"background_color": "black"
}
}
One may access the enable_autocomplete
field by supplying superclient
as
the namespace
argument and settings.console.enable_autocomplete
or
["settings", "console", "enable_autocomplete"]
as the key
argument for
the request. The entire settings object could be accessed by providing
settings
or ["settings"]
as the key
argument. The entire namespace
may be read by omitting the key
argument, however as explained below it
is not possible to modify a namespace without specifying a key.
List namespaces
Lists all available namespaces.
HTTP request:
GET /server/database/list
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.database.list",
"id": 8694
}
Returns:
An object containing an array of namespaces in the following format:
{
"namespaces": [
"gcode_metadata",
"history",
"moonraker",
"test_namespace"
]
}
Get Database Item
Retrieves an item from a specified namespace. The key
argument may be
omitted, in which case an object representing the entire namespace will
be returned in the value
field. If the key
is provided and does not
exist in the database an error will be returned.
HTTP request:
GET /server/database/item?namespace={namespace}&key={key}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.database.get_item",
"params": {
"namespace": "{namespace}",
"key": "{key}"
},
"id": 5644
}
Returns:
An object containing the requested namespace
, key
, and value
.
{
"namespace": "moonraker",
"key": "file_manager.metadata_version",
"value": 2
}
Add Database Item
Inserts an item into the database. If the namespace
does not exist
it will be created. If the key
specifies a nested field, all parents
will be created if they do not exist. If the key exists it will be
overwritten with the provided value
. The key
parameter must be provided,
as it is not possible to assign a value directly to a namespace.
HTTP request:
POST /server/database/item?namespace={namespace}&key={key}value={value}`
!!! note
If the value
is not a string type, the value
argument must
provide a type hint. Alternatively,
arguments may be passed via the request body in JSON format. For
example:
POST /server/database/item
Content-Type: application/json
{
"namespace": "my_client",
"key": "settings.some_count",
"value": 100
}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.database.post_item",
"params": {
"namespace": "{namespace}",
"key": "{key}",
"value": 100
},
"id": 4654
}
Returns:
An object containing the inserted namespace
, key
, and value
.
{
"namespace": "test",
"key": "settings.some_count",
"value": 9001
}
Delete Database Item
Deletes an item from a namespace
at the specified key
. If the key does not
exist in the namespace an error will be returned. If the deleted item results
in an empty namespace, the namespace will be removed from the database.
HTTP request:
DELETE /server/database/item?namespace={namespace}&key={key}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.database.delete_item",
"params": {
"namespace": "{namespace}",
"key": "{key}"
},
"id": 4654
}
Returns:
An object containing the namespace
, key
, and value
of the
deleted item.
{
"namespace": "test",
"key": "settings.some_count",
"value": 9001
}
Job Queue APIs
The following enpoints may be used to manage Moonraker's job queue. Note that Moonraker's Job Queue is impelemented as a FIFO queue and it may contain multiple references to the same job.
!!! Note
All filenames provided to and returned by these endpoints are relative to
the gcodes
root.
Retrieve the job queue status
Retrieves the current state of the job queue
HTTP request:
GET /server/job_queue/status
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.job_queue.status",
"id": 4654
}
Returns:
The current state of the job queue:
{
"queued_jobs": [
{
"filename": "job1.gcode",
"job_id": "0000000066D99C90",
"time_added": 1636151050.7666452,
"time_in_queue": 21.89680004119873
},
{
"filename": "job2.gcode",
"job_id": "0000000066D991F0",
"time_added": 1636151050.7766452,
"time_in_queue": 21.88680004119873
},
{
"filename": "subdir/job3.gcode",
"job_id": "0000000066D99D80",
"time_added": 1636151050.7866452,
"time_in_queue": 21.90680004119873
}
],
"queue_state": "ready"
}
Below is a description of the returned fields:
queued_jobs
: an array of objects representing each queued job. Each object contains thefilename
of the enqueued job and a uniquejob_id
generated for each job. Thejob_id
is a 64-bit Hexadecimal string value. On 32-bit systems the most significant bits will always contain zeros. Items are ordered by the time they were queued, the first item will be the next job loaded.queue_state
: The current state of the queue. Can be one of the following:ready
: The queue is active and will load the next job upon completion of the current jobloading
: The queue is currently loading the next job. If the user specified ajob_transition_delay
and/orjob_transition_gcode
, the queue will remain in theloading
state until both are completed or an error is encountered.starting
: The queue enters this state after theloading
phase is complete before attempting to start the job.paused
: The queue is currently paused and will not load the next job upon completion of the current job. The queue will enter thepaused
state if an error is encountered during theloading
orstarting
phases, or if the user pauses the queue through the provided endpoint.
time_added
: The time (in Unix Time) the job was added to the queuetime_in_queue
: The cumulative amount of time (in seconds) the job has been pending in the queue
Enqueue a job
Adds a job, or an array of jobs, to the end of the job queue. The same filename may be specified multiple times to queue a job that repeats. When multiple jobs are specfied they will be enqued in the order they are received.
!!! Note The request will be aborted and return an error if any of the supplied files do not exist.
HTTP request:
POST /server/job_queue/job?filenames=job1.gcode,job2.gcode,subdir/job3.gocde
!!! Note
Multiple jobs should be comma separated as shown above.
Alternatively filenames
maybe be specified as a json object
in the body of the request.
POST /server/job_queue/job
Content-Type: application/json
{
"filenames": [
"job1.gcode",
"job2.gcode",
"subdir/job3.gcode"
]
}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.job_queue.post_job",
"params": {
"filenames": [
"job1.gcode",
"job2.gcode",
"subir/job3.gcode"
]
},
"id": 4654
}
Returns:
The current state of the job queue:
{
"queued_jobs": [
{
"filename": "job1.gcode",
"job_id": "0000000066D99C90",
"time_added": 1636151050.7666452,
"time_in_queue": 21.89680004119873
},
{
"filename": "job2.gcode",
"job_id": "0000000066D991F0",
"time_added": 1636151050.7766452,
"time_in_queue": 21.88680004119873
},
{
"filename": "subdir/job3.gcode",
"job_id": "0000000066D99D80",
"time_added": 1636151050.7866452,
"time_in_queue": 21.90680004119873
}
],
"queue_state": "ready"
}
Remove a Job
Removes one or more jobs from the queue.
!!! Note Unlike the POST version of this method, it is not necessary that all job ids exist. If any supplied job id does not exist in the queue it will be silently ignored. Clients can verify the contents of the queue via the return value.
HTTP request:
DELETE /server/job_queue/job?job_ids=0000000066D991F0,0000000066D99D80
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.job_queue.delete_job",
"params": {
"job_ids": [
"0000000066D991F0".
"0000000066D99D80"
]
},
"id": 4654
}
!!! Tip
Alternatively all=true
("all": true
for JSON-RPC) may specified
to clear the job queue.
Returns:
The current state of the job queue:
{
"queued_jobs": [
{
"filename": "job1.gcode",
"job_id": "0000000066D99C90",
"time_added": 1636151050.7666452,
"time_in_queue": 21.89680004119873
}
],
"queue_state": "ready"
}
Pause the job queue
Sets the job queue state to "pause", which prevents the next job in the queue from loading after an job in progress is complete.
!!! Note
If the queue is paused while the queue is in the loading
state
the load will be aborted.
HTTP request:
POST /server/job_queue/pause
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.job_queue.pause",
"id": 4654
}
Returns:
The current state of the job queue:
{
"queued_jobs": [
{
"filename": "job1.gcode",
"job_id": "0000000066D99C90",
"time_added": 1636151050.7666452,
"time_in_queue": 21.89680004119873
},
{
"filename": "job2.gcode",
"job_id": "0000000066D991F0",
"time_added": 1636151050.7766452,
"time_in_queue": 21.88680004119873
},
{
"filename": "subdir/job3.gcode",
"job_id": "0000000066D99D80",
"time_added": 1636151050.7866452,
"time_in_queue": 21.90680004119873
}
],
"queue_state": "paused"
}
Start the job queue
Starts the job queue. If Klipper is ready to start a print the next job in the queue will be loaded. Otherwise the queue will be put into the "ready" state, enabling automatic transition after the next completed print.
HTTP request:
POST /server/job_queue/start
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.job_queue.start",
"id": 4654
}
Returns:
The current state of the job queue:
{
"queued_jobs": [
{
"filename": "job1.gcode",
"job_id": "0000000066D99C90",
"time_added": 1636151050.7666452,
"time_in_queue": 21.89680004119873
},
{
"filename": "job2.gcode",
"job_id": "0000000066D991F0",
"time_added": 1636151050.7766452,
"time_in_queue": 21.88680004119873
},
{
"filename": "subdir/job3.gcode",
"job_id": "0000000066D99D80",
"time_added": 1636151050.7866452,
"time_in_queue": 21.90680004119873
}
],
"queue_state": "loading"
}
Announcement APIs
The following endpoints are available to manage announcements. See the appendix for details on how announcements work and recommendations for your implementation.
List announcements
Retreives a list of current announcements. The include_dismissed
argument is optional and defaults to true
. If set to false
dismissed entries will be omitted from the return value.
HTTP request:
GET /server/announcements/list?include_dismissed=false
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.announcements.list",
"params": {
"include_dismissed": false
},
"id": 4654
}
Returns:
The current list of announcements, in descending order (newest to oldest)
sorted by date
and a list of feeds Moonraker is currently subscribed to:
{
{
"entries": [
{
"entry_id": "arksine/moonlight/issue/3",
"url": "https://github.com/Arksine/moonlight/issues/3",
"title": "Test announcement 3",
"description": "Test Description [with a link](https://moonraker.readthedocs.io).",
"priority": "normal",
"date": 1647459219,
"dismissed": false,
"date_dismissed": null,
"dismiss_wake": null,
"source": "moonlight",
"feed": "moonlight"
},
{
"entry_id": "arksine/moonlight/issue/2",
"url": "https://github.com/Arksine/moonlight/issues/2",
"title": "Announcement Test Two",
"description": "This is a high priority announcement. This line is included in the description.",
"priority": "high",
"date": 1646855579,
"dismissed": false,
"date_dismissed": null,
"dismiss_wake": null,
"source": "moonlight",
"feed": "moonlight"
},
{
"entry_id": "arksine/moonlight/issue/1",
"url": "https://github.com/Arksine/moonlight/issues/1",
"title": "Announcement Test One",
"description": "This is the description. Anything here should appear in the announcement, up to 512 characters.",
"priority": "normal",
"date": 1646854678,
"dismissed": false,
"date_dismissed": null,
"dismiss_wake": null,
"source": "moonlight",
"feed": "moonlight"
},
{
"entry_id": "arksine/moonraker/issue/349",
"url": "https://github.com/Arksine/moonraker/issues/349",
"title": "PolicyKit warnings; unable to manage services, restart system, or update packages",
"description": "This announcement is an effort to get ahead of a coming change that will certainly result in issues. PR #346 has been merged, and with it are some changes to Moonraker's default behavior.",
"priority": "normal",
"date": 1643392406,
"dismissed": false,
"source": "moonlight",
"feed": "Moonraker"
}
],
"feeds": [
"moonraker",
"klipper",
"moonlight"
]
}
}
Update announcements
Requests that Moonraker check for announcement updates. This is generally not required in production, as Moonraker will automatically check for updates every 30 minutes. However, during development this endpoint is useful to force an update when it is necessary to perform integration tests.
HTTP request:
POST /server/announcements/update
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.announcements.update",
"id": 4654
}
Returns:
The current list of announcements, in descending order (newest to oldest)
sorted by date
, and a modified
field that contains a boolean value
indicating if the update resulted in a change:
{
"entries": [
{
"entry_id": "arksine/moonraker/issue/349",
"url": "https://github.com/Arksine/moonraker/issues/349",
"title": "PolicyKit warnings; unable to manage services, restart system, or update packages",
"description": "This announcement is an effort to get ahead of a coming change that will certainly result in issues. PR #346 has been merged, and with it are some changes to Moonraker's default behavior.",
"priority": "normal",
"date": 1643392406,
"dismissed": false,
"source": "moonlight",
"feed": "Moonraker"
},
{
"entry_id": "arksine/moonlight/issue/1",
"url": "https://github.com/Arksine/moonlight/issues/1",
"title": "Announcement Test One",
"description": "This is the description. Anything here should appear in the announcement, up to 512 characters.",
"priority": "normal",
"date": 1646854678,
"dismissed": true,
"source": "moonlight",
"feed": "Moonlight"
},
{
"entry_id": "arksine/moonlight/issue/2",
"url": "https://github.com/Arksine/moonlight/issues/2",
"title": "Announcement Test Two",
"description": "This is a high priority announcement. This line is included in the description.",
"priority": "high",
"date": 1646855579,
"dismissed": false,
"source": "moonlight",
"feed": "Moonlight"
},
{
"entry_id": "arksine/moonlight/issue/3",
"url": "https://github.com/Arksine/moonlight/issues/3",
"title": "Test announcement 3",
"description": "Test Description [with a link](https://moonraker.readthedocs.io).",
"priority": "normal",
"date": 1647459219,
"dismissed": false,
"source": "moonlight",
"feed": "Moonlight"
}
],
"modified": false
}
Dismiss an announcement
Sets the dismiss flag of an announcement to true
. The entry_id
field is required. The entry_id
contains forward slashes so remember
to escape the ID if including it in the query string of an HTTP request.
HTTP request:
POST /server/announcements/dismiss?entry_id=arksine%2Fmoonlight%2Fissue%2F1&wake_time=600
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.announcements.dismiss",
"params": {
"entry_id": "arksine/moonlight/issue/1",
"wake_time": 600
},
"id": 4654
}
Parameters:
entry_id
: The entry identifier. This field may contain forward slashes so it should be url escaped when placed in the query string of an http request. This parameter is required.wake_time
: The time, in seconds, in which the entry'sdismissed
state will revert to false. This parameter is optional, if omitted the entry will be dismissed indefinitely.
Returns:
The entry_id
of the dismissed entry:
{
"entry_id": "arksine/moonlight/issue/1"
}
Add an announcement feed
Specifies a new feed for Moonraker's announcements
component to query
in addition to moonraker
, klipper
, and feeds configured in
moonraker.conf
.
HTTP request:
POST /server/announcements/feed?name=my_feed
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.announcements.post_feed",
"params": {
"name": "my_feed"
},
"id": 4654
}
Parameters:
name
: The name of the new feed. This parameter is required.
Returns:
The name of the new feed and the action taken. The action
will be added
if a new feed added, or skipped
if the feed already exists.
{
"feed": "my_feed",
"action": "added"
}
Remove an announcement feed
Removes a subscribed feed. Only feeds previously subscribed to using
the add feed API may be removed. Feeds
configured in moonraker.conf
may not be removed.
HTTP request:
DELETE /server/announcements/feed?name=my_feed
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.announcements.delete_feed",
"params": {
"name": "my_feed"
},
"id": 4654
}
Parameters:
name
: The name of the new feed to remove. This parameter is required.
Returns:
The name of the new feed and the action taken. The action
will be
removed
if the operation was successful.
{
"feed": "my_feed",
"action": "removed"
}
Update Manager APIs
The following endpoints are available when the [update_manager]
component has
been configured:
Get update status
Retrieves the current state of each "package" available for update. Typically
this will consist of information regarding moonraker
, klipper
, system
packages, along with configured clients. If moonraker has not yet received
information from Klipper then its status will be omitted. One may request that
the update info be refreshed by setting the refresh
argument to true
. Note
that the refresh
argument is ignored if an update is in progress or if a print
is in progress. In these cases the current status will be returned immediately
and no refresh will take place. If the refresh
argument is omitted its value
defaults to false
.
HTTP request:
GET /machine/update/status?refresh=false
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.update.status",
"params": {
"refresh": false
},
"id": 4644
}
Returns:
Status information for each update package. Note that mainsail
and fluidd
are present as clients configured in moonraker.conf
{
"busy": false,
"github_rate_limit": 60,
"github_requests_remaining": 57,
"github_limit_reset_time": 1615836932,
"version_info": {
"system": {
"package_count": 4,
"package_list": [
"libtiff5",
"raspberrypi-sys-mods",
"rpi-eeprom-images",
"rpi-eeprom"
]
},
"moonraker": {
"channel": "dev",
"debug_enabled": true,
"need_channel_update": false,
"is_valid": true,
"configured_type": "git_repo",
"info_tags": [],
"detected_type": "git_repo",
"remote_alias": "arksine",
"branch": "master",
"owner": "?",
"repo_name": "moonraker",
"version": "v0.7.1-364",
"remote_version": "v0.7.1-364",
"current_hash": "ecfad5cff15fff1d82cb9bdc64d6b548ed53dfaf",
"remote_hash": "ecfad5cff15fff1d82cb9bdc64d6b548ed53dfaf",
"is_dirty": false,
"detached": true,
"commits_behind": [],
"git_messages": [],
"full_version_string": "v0.7.1-364-gecfad5c",
"pristine": true
},
"mainsail": {
"name": "mainsail",
"owner": "mainsail-crew",
"version": "v2.1.1",
"remote_version": "v2.1.1",
"configured_type": "web",
"channel": "stable",
"info_tags": [
"desc=Mainsail Web Client",
"action=some_action"
]
},
"fluidd": {
"name": "fluidd",
"owner": "cadriel",
"version": "?",
"remote_version": "v1.16.2",
"configured_type": "web_beta",
"channel": "beta",
"info_tags": []
},
"klipper": {
"channel": "dev",
"debug_enabled": true,
"need_channel_update": false,
"is_valid": true,
"configured_type": "git_repo",
"info_tags": [],
"detected_type": "git_repo",
"remote_alias": "origin",
"branch": "master",
"owner": "Klipper3d",
"repo_name": "klipper",
"version": "v0.10.0-1",
"remote_version": "v0.10.0-41",
"current_hash": "4c8d24ae03eadf3fc5a28efb1209ce810251d02d",
"remote_hash": "e3cbe7ea3663a8cd10207a9aecc4e5458aeb1f1f",
"is_dirty": false,
"detached": false,
"commits_behind": [
{
"sha": "e3cbe7ea3663a8cd10207a9aecc4e5458aeb1f1f",
"author": "Kevin O'Connor",
"date": "1644534721",
"subject": "stm32: Clear SPE flag on a change to SPI CR1 register",
"message": "The stm32 specs indicate that the SPE bit must be cleared before\nchanging the CPHA or CPOL bits.\n\nReported by @cbc02009 and @bigtreetech.\n\nSigned-off-by: Kevin O'Connor <kevin@koconnor.net>",
"tag": null
},
{
"sha": "99d55185a21703611b862f6ce4b80bba70a9c4b5",
"author": "Kevin O'Connor",
"date": "1644532075",
"subject": "stm32: Wait for transmission to complete before returning from spi_transfer()",
"message": "It's possible for the SCLK pin to still be updating even after the\nlast byte of data has been read from the receive pin. (In particular\nin spi mode 0 and 1.) Exiting early from spi_transfer() in this case\ncould result in the CS pin being raised before the final updates to\nSCLK pin.\n\nAdd an additional wait at the end of spi_transfer() to avoid this\nissue.\n\nSigned-off-by: Kevin O'Connor <kevin@koconnor.net>",
"tag": null
},
],
"git_messages": [],
"full_version_string": "v0.10.0-1-g4c8d24ae-shallow",
"pristine": true
}
}
}
Below is an explanation for each field:
busy
: set to true if an update is in progress. Moonraker will not allow concurrent updates.github_rate_limit
: the maximum number of github API requests the user currently is allowed. An unathenticated user typically has 60 requests per hour.github_requests_remaining
: the number of API request the user currently has remaining.github_limit_reset_time
: the time when the rate limit will reset, reported as seconds since the epoch (aka Unix Time).
The moonraker
, klipper
packages, along with and clients configured
as applications have the following fields:
configured_type
: the application type configured by the userdetected_type
: the applicaiton type as detected by Moonraker.channel
: the currently configured update channel. For Moonraker and Klipper this is set in the[update_manager]
configuration. For clients the channel is determined by the configured typeneed_channel_update
: This will be set totrue
if Moonraker has detected that a channel swap is necessary (ie: the configured type does not match the detected type). The channel swap will be performed on the next update.pristine
: Forzip
andzip_beta
types this is set totrue
if an applications source checksum matches the one generated when the app was built. This value will be set to the opposite of "dirty" for git repos. Note that a zip application can still be updated if the repo is not pristine.owner
: the owner of the repo / applicationbranch
: the name of the current git branch. This should typically be "master".remote_alias
: the alias for the remote. This should typically be "origin".version
: abbreviated version of the current repo on diskremote_version
: abbreviated version of the latest available updatefull_version_string
: The complete version string of the current repo.current_hash
: hash of the most recent commit on diskremote_hash
: hash of the most recent commit pushed to the remoteis_valid
: true if installation is a valid git repo on the master branch and an "origin" set to the official remote. Forzip
andzip_beta
types this will report false if Moonraker is unable to fetch the current repo state from GitHub.is_dirty
: true if the repo has been modified. This will always be false forzip
andzip_beta
types.detached
: true if the repo is currently in a detached state. Forzip
andzip_beta
types it is considered detached if the local release info does not match what is present on the remote.debug_enabled
: True whenenable_repo_debug
has been configured. This will bypass repo validation allowing detached updates, and updates from a remote/branch other than than the primary (typically origin/master).commits_behind
: A list of commits behind. Up to 30 "untagged" commits will be reported. Moonraker checks the last 100 commits for tags, any commits beyond the last 30 with a tag will also be reported.git_messages
: If a repo is in the "invalid" state this field will hold a list of string messages containing the output of the last failed git command. Note that it is possible for a git command to fail without providing output (for example, it may become non-responsive and time out), so it is possible for this field to be an empty list when the repo is invalid.info_tags
: These are tags defined in the[update_manager client_name]
configuration for each client. Client developers my define what tags, if any, users will configure. They can choose to use those tags to display information or perform an additional action after an update if necessary.
Web clients have the following fields:
channel
: channel to fetch updates fromconfigured_type
: will beweb
orweb_beta
name
: name of the configured clientowner
: the owner of the clientversion
: version of the installed client.remote_version
: version of the latest release published to GitHubinfo_tags
: These are tags defined in the[update_manager client_name]
configuration for each client. Client developers my define what tags, if any, users will configure. They can choose to use those tags to display information or perform an additional action after an update if necessary.
The system
package has the following fields:
package_count
: the number of system packages available for updatepackage_list
: an array containing the names of packages available for update
Perform a full update
Attempts to update all configured items in Moonraker. Updates are performed in the following order:
system
if enabled- All configured clients
- Klipper
- Moonraker
HTTP request:
POST /machine/update/full
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.update.full",
"id": 4645
}
Returns:
ok
when complete
Update Moonraker
Pulls the most recent version of Moonraker from GitHub and restarts the service. If an update is requested while a print is in progress then this request will return an error.
HTTP request:
POST /machine/update/moonraker
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.update.moonraker",
"id": 4645
}
Returns:
ok
when complete
Update Klipper
Pulls the most recent version of Klipper from GitHub and restarts the service. If an update is requested while a print is in progress then this request will return an error.
HTTP request:
POST /machine/update/klipper
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.update.klipper",
"id": 5745
}
Returns:
ok
when complete
Update Client
If one more more [update_manager client client_name]
sections have
been configured this endpoint can be used to install the most recently
published release of the client. If an update is requested while a
print is in progress then this request will return an error. The
name
argument is requred, it's value should match the client_name
of the configured section.
HTTP request:
POST /machine/update/client?name={client_name}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.update.client",
"params": {
"name": "client_name"
},
"id": 8546
}
Returns:
ok
when complete
Update System Packages
Upgrades system packages. Currently only apt-get
is supported.
If an update is requested while a print is in progress then this request
will return an error.
HTTP request:
POST /machine/update/system
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.update.system",
"id": 4564
}
Returns:
ok
when complete
Recover a corrupt repo
On ocassion a git command may fail resulting in a repo in a
dirty or invalid state. When this happens it is possible
to recover. The name
argument must specify the name of
the repo to recover, it must be of a git repo type. There are two
methods of recovery, the hard
argument determines which method
is used:
hard == true
: Moonraker will remove the old directory entirely. It will then attempt to recover withrsync
by restoring a backup of a recent valid repo.hard == false
: Will rungit clean -f -d
followed bygit reset --hard {remote}/{branch}
. This is useful for recovering dirty repos that are valid. It is possible that this will work on an invalid repo, however it will not work on a corrupt repo.
The hard
argument defaults to false
.
HTTP request:
POST /machine/update/recover?name=moonraker&hard=false
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.update.recover",
"params": {
"name": "moonraker",
"hard": false
},
"id": 4564
}
Returns:
ok
when complete
Power APIs
The APIs below are available when the [power]
component has been configured.
Get Device List
HTTP request:
GET /machine/device_power/devices
JSON-RPC request:
{
"jsonrpc": "2.0",
"method":"machine.device_power.devices",
"id": 5646
}
Returns:
An array of objects containing info for each configured device.
{
"devices": [
{
"device": "green_led",
"status": "off",
"locked_while_printing": true,
"type": "gpio"
},
{
"device": "printer",
"status": "off",
"locked_while_printing": false,
"type": "tplink_smartplug"
}
]
}
Get Device Status
Returns the status for a single configured device.
HTTP request:
GET /machine/device_power/device?device=green_led
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.device_power.get_device",
"params": {
"device": "green_led"
},
"id": 4564
}
Returns:
An object containing power state for the requested device:
{
"green_led": "off"
}
Set Device State
Toggle, turn on, or turn off a specified device.
HTTP request:
POST /machine/device_power/device?device=green_led&action=on
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.device_power.post_device",
"params": {
"device": "green_led",
"action": "on"
},
"id": 4564
}
!!! note
The action
argument may be on
, off
, or toggle
. Any
other value will result in an error.
Returns:
An object containing new power state for the requested device:
{
"green_led": "off"
}
Get Batch Device Status
Get power status for the requested devices. At least one device must be specified.
HTTP request:
GET /machine/device_power/status?dev_one&dev_two
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.device_power.status",
"params": {
"dev_one":null,
"dev_two": null
},
"id": 4564
}
Returns:
An object containing power state for each requested device:
{
"green_led": "off",
"printer": "off"
}
Batch Power On Devices
Power on the requested devices. At least one device must be specified.
HTTP request:
POST /machine/device_power/on?dev_one&dev_two
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.device_power.on",
"params": {
"dev_one":null,
"dev_two": null
},
"id": 4564
}
An object containing power state for each requested device:
{
"green_led": "on",
"printer": "on"
}
Batch Power Off Devices
Power off the requested devices. At least one device must be specified.
HTTP request:
POST /machine/device_power/off?dev_one&dev_two
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "machine.device_power.off",
"params": {
"dev_one":null,
"dev_two": null
},
"id": 4564
}
An object containing power state for each requested device:
{
"green_led": "off",
"printer": "off"
}
OctoPrint API emulation
Partial support of OctoPrint API is implemented with the purpose of allowing uploading of sliced prints to a moonraker instance. Currently we support Slic3r derivatives and Cura with Cura-OctoPrint.
Version information
HTTP request:
GET /api/version
JSON-RPC request: Not Available
Returns:
An object containing simulated OctoPrint version information
{
"server": "1.5.0",
"api": "0.1",
"text": "OctoPrint (Moonraker v0.3.1-12)"
}
Server status
HTTP request:
GET /api/server
JSON-RPC request: Not Available
Returns:
An object containing simulated OctoPrint server status
{
"server": "1.5.0",
"safemode": "settings"
}
Login verification & User information
HTTP request:
GET /api/login
JSON-RPC request: Not Available
Returns:
An object containing stubbed OctoPrint login/user verification
{
"_is_external_client": false,
"_login_mechanism": "apikey",
"name": "_api",
"active": true,
"user": true,
"admin": true,
"apikey": null,
"permissions": [],
"groups": ["admins", "users"],
}
Get settings
HTTP request:
GET /api/settings
JSON-RPC request: Not Available
Returns:
An object containing stubbed OctoPrint settings. The webcam route is hardcoded to Fluidd/Mainsail default path. We say we have the UFP plugin installed so that Cura-OctoPrint will upload in the preferred UFP format.
{
"plugins": {
"UltimakerFormatPackage": {
"align_inline_thumbnail": false,
"inline_thumbnail": false,
"inline_thumbnail_align_value": "left",
"inline_thumbnail_scale_value": "50",
"installed": true,
"installed_version": "0.2.2",
"scale_inline_thumbnail": false,
"state_panel_thumbnail": true
}
},
"feature": {
"sdSupport": false,
"temperatureGraph": false
},
"webcam": {
"flipH": false,
"flipV": false,
"rotate90": false,
"streamUrl": "/webcam/?action=stream",
"webcamEnabled": true
}
}
OctoPrint File Upload
HTTP request:
POST /api/files/local
JSON-RPC request: Not Available
Alias for Moonrakers file upload API.
Get Job status
HTTP request:
GET /api/job
JSON-RPC request: Not Available
Returns:
An object containing stubbed OctoPrint Job status
{
"job": {
"file": {"name": null},
"estimatedPrintTime": null,
"filament": {"length": null},
"user": null
},
"progress": {
"completion": null,
"filepos": null,
"printTime": null,
"printTimeLeft": null,
"printTimeOrigin": null
},
"state": "Offline"
}
Get Printer status
HTTP request:
GET /api/printer
JSON-RPC request: Not Available
Returns:
An object containing OctoPrint Printer status
{
"temperature": {
"tool0": {
"actual": 22.25,
"offset": 0,
"target": 0
},
"bed": {
"actual": 22.25,
"offset": 0,
"target": 0
}, ...<additional heaters>
},
"state": {
"text": "state",
"flags": {
"operational": true,
"paused": false,
"printing": false,
"cancelling": false,
"pausing": false,
"error": false,
"ready": false,
"closedOrError": false
}
}
}
Send GCode command
HTTP request:
POST /api/printer/command
Content-Type: application/json
{
"commands": ["G28"]
}
JSON-RPC request: Not Available
Returns:
An empty JSON object
{}
List Printer profiles
HTTP request:
GET /api/printerprofiles
JSON-RPC request: Not Available
Returns:
An object containing simulates OctoPrint Printer profile
{
"profiles": {
"_default": {
"id": "_default",
"name": "Default",
"color": "default",
"model": "Default",
"default": true,
"current": true,
"heatedBed": true,
"heatedChamber": false
}
}
}
History APIs
The APIs below are avilable when the [history]
component has been configured.
Get job list
HTTP request:
GET /server/history/list?limit=50&start=50&since=1&before=5&order=asc
JSON-RPC request:
{
"jsonrpc": "2.0",
"method":"server.history.list",
"params":{
"limit": 50,
"start": 10,
"since": 464.54,
"before": 1322.54,
"order": "asc"
},
"id": 5656
}
All arguments are optional. Arguments are as follows:
start
Record number to start from (i.e. 10 would start at the 10th print)limit
Maximum Number of prints to return (default: 50)before
All jobs before this UNIX timestampsince
All jobs after this UNIX timestamporder
Define return orderasc
ordesc
(default)
Returns:
An array of requsted historical jobs:
{
"count": 1,
"jobs": [
{
"job_id": "000001",
"exists": true,
"end_time": 1615764265.6493807,
"filament_used": 7.83,
"filename": "test/history_test.gcode",
"metadata": {
// Object containing metadata at time of job
},
"print_duration": 18.37201827496756,
"status": "completed",
"start_time": 1615764496.622146,
"total_duration": 18.37201827496756
},
]
}
Get job totals
HTTP request:
GET /server/history/totals
JSON-RPC request:
{
"jsonrpc": "2.0",
"method":"server.history.totals",
"id": 5656
}
Returns:
An object containing the following total job statistics:
{
"job_totals": {
"total_jobs": 3,
"total_time": 11748.077333278954,
"total_print_time": 11348.794790096988,
"total_filament_used": 11615.718840001999,
"longest_job": 11665.191012736992,
"longest_print": 11348.794790096988
}
}
Reset totals
Resets the persistent "job totals" to zero.
HTTP request:
POST /server/history/reset_totals
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.history.reset_totals",
"id": 5534
}
Returns:
The totals prior to the reset:
```json
{
"last_totals": {
"total_jobs": 3,
"total_time": 11748.077333278954,
"total_print_time": 11348.794790096988,
"total_filament_used": 11615.718840001999,
"longest_job": 11665.191012736992,
"longest_print": 11348.794790096988
}
}
Get a single job
HTTP request:
GET /server/history/job?uid=<id>
JSON-RPC request:
{
"jsonrpc": "2.0",
"method":"server.history.get_job",
"params":{"uid": "{uid}"},
"id": 4564
}
Returns:
Data associated with the job ID in the following format:
{
"job": {
"job_id": "000001",
"exists": true,
"end_time": 1615764265.6493807,
"filament_used": 7.83,
"filename": "test/history_test.gcode",
"metadata": {
// Object containing metadata at time of job
},
"print_duration": 18.37201827496756,
"status": "completed",
"start_time": 1615764496.622146,
"total_duration": 18.37201827496756
}
}
Delete job
HTTP request:
DELETE /server/history/job?uid=<id>
JSON-RPC request:
{
"jsonrpc": "2.0",
"method": "server.history.delete_job",
"params":{
"uid": "{uid}"
},
"id": 5534
}
!!! tip
It is possible to replace the uid
argument with all=true
to delete all jobs in the history database.
Returns:
An array of deleted job ids
[
"000000",
"000001",
]
MQTT APIs
The following API is available when [mqtt]
has been configured.
!!! Note
These requests are not available over the mqtt
transport as they
are redundant. MQTT clients can publish and subscribe to
topics directly.
Publish a topic
HTTP request:
POST /server/mqtt/publish
Content-Type: application/json
{
"topic": "home/test/pub",
"payload": "hello",
"qos": 0,
"retain": false,
"timeout": 5
}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method":"server.mqtt.publish",
"params":{
"topic": "home/test/pub",
"payload": "hello",
"qos": 0,
"retain": false,
"timeout": 5
},
"id": 4564
}
Only the topic
parameter is required. Below is an explanation for
each paramater:
topic
: The topic to publish.payload
: Payload to send with the topic. May be a boolean, float, integer, string, object, or array. All values are converted to strings prior to publishing. Objects and Arrays are JSON encoded. If omitted an empty payload is sent.qos
: QOS level to use when publishing the topic. Must be an integer value from 0 to 2. If omitted the system configured default is used.retain
: If set totrue
the MQTT broker will retain the payload of this request. Note that only the mostly recently tagged payload is retained. When other clients first subscribe to the topic they immediately recieve the retained message. The default isfalse
.timeout
: A float value in seconds. By default requests with QoS levels of 1 or 2 will block until the Broker acknowledges confirmation. This option applies a timeout to the request, returning a 504 error if the timeout is exceeded. Note that the topic will still be published if the QoS level is 1 or 2.
!!! tip
To clear a retained value of a topic, publish the topic with an empty
payload and retain
set to true
.
Returns:
The published topic:
{
"topic": "home/test/pub"
}
Subscribe to a topic
HTTP request:
POST /server/mqtt/subscribe
Content-Type: application/json
{
"topic": "home/test/sub",
"qos": 0,
"timeout": 5
}
JSON-RPC request:
{
"jsonrpc": "2.0",
"method":"server.mqtt.subscribe",
"params":{
"topic": "home/test/sub",
"qos": 0,
"timeout": 5
},
"id": 4564
}
Only the topic
parameter is required. Below is an explanation for
each paramater:
topic
: The topic to subscribe. Note that wildcards may not be used.qos
: QOS level to use when subscribing to the topic. Must be an integer value from 0 to 2. If omitted the system configured default is used.timeout
: A float value in seconds. By default requests will block indefinitely until a response is received. This option applies a timeout to the request, returning a 504 error if the timeout is exceeded. The subscription will be removed after a timeout.
!!! note If the topic was previously published with a retained payload this request will return with the retained value.
Returns:
The subscribed topic and its payload:
{
"topic": "home/test/pub",
"payload": "test"
}
If the payload is json encodable it will be returned as an object or array. Otherwise it will be a string.
Websocket notifications
Printer generated events are sent over the websocket as JSON-RPC 2.0 notifications. These notifications are sent to all connected clients in the following format:
{
"jsonrpc": "2.0",
"method": "{event method name}"
}
OR
{
"jsonrpc": "2.0",
"method": "{event method name}",
"params": [<event parameter>]
}
If a notification has parameters, the params
value will always be
wrapped in an array as directed by the JSON-RPC standard. Currently
all notifications available are broadcast with either no parameters
or a single parameter.
Gcode Response
All of Klippy's gcode responses are forwarded over the websocket. They arrive as a "gcode_response" notification:
{
"jsonrpc": "2.0",
"method": "notify_gcode_response",
"params": ["response message"]
}
Subscriptions
Status Subscriptions arrive as a "notify_status_update" notification:
{
"jsonrpc": "2.0",
"method": "notify_status_update",
"params": [{<status object>}, <eventtime>]
}
The structure of the status object
is identical to the structure that is
returned from an object query's
status
field.
The eventtime
is a timestamp generated by Klipper when
the update was originally pushed. This timestamp is a float value,
relative to Klipper's monotonic clock.
Klippy Ready
Notify clients when Klippy has reported a ready state
{
"jsonrpc": "2.0",
"method": "notify_klippy_ready"
}
Klippy Shutdown
Notify clients when Klippy has reported a shutdown state
{
"jsonrpc": "2.0",
"method": "notify_klippy_shutdown"
}
Klippy Disconnected
Notify clients when Moonraker's connection to Klippy has terminated
{
"jsonrpc": "2.0",
"method": "notify_klippy_disconnected"
}
File List Changed
When a client makes a change to a file or directory in a registered
root
(via upload, delete, move, etc) a notification is broadcast
to alert all connected clients of the change:
{
"jsonrpc": "2.0",
"method": "notify_filelist_changed",
"params": [
{
"action": "{action}",
"item": {
"path": "{file or directory path}",
"root": "{root}",
"size": 46458,
"modified": 545465
},
"source_item": {
"path": "{file or directory path}",
"root": "{root_name}"
}
}
]
}
The source_item
field is only present for move_item
and
copy_item
actions. The action
field will be set
to one of the following values:
create_file
create_dir
delete_file
delete_dir
move_file
move_dir
modify_file
root_update
Most of the above actions are self explanatory. The root_update
notification is sent when a root
folder has changed its location,
for example when a user configures a different gcode file path
in Klipper.
Update Manager Response
The update manager will send asyncronous messages to the client during an update:
{
"jsonrpc": "2.0",
"method": "notify_update_response",
"params": [
{
"application": "{app_name}",
"proc_id": 446461,
"message": "Update Response Message",
"complete": false
}
]
}
The fields reported in the response are as follows:
- The
application
field contains the name of application currently being updated. Generally this will be either "moonraker", "klipper", "system", or "client". - The
proc_id
field contains a unique id associated with the current update process. This id is generated for each update request. - The
message
field contains an asyncronous message sent during the update process. - The
complete
field is set to true on the final message sent during an update, indicating that the update completed successfully. Otherwise it will be false.
Update Manager Refreshed
The update manager periodically auto refreshes the state of each application it is tracking. After an auto refresh has completed the following notification is broadcast:
{
"jsonrpc": "2.0",
"method": "notify_update_refreshed",
"params": [{update_info}]}
Where update_info
is an object that matches the response from an
update status request.
CPU Throttled
If the system supports throttled CPU monitoring Moonraker will send the following notification when it detects a change to the current throttled state:
{
"jsonrpc": "2.0",
"method": "notify_cpu_throttled",
"params": [{throttled_state}]
}
Where throttled_state
is an object that matches the throttled_state
field
in the response from a Moonraker process stats
request. It is possible for clients to receive this notification multiple times
if the system repeatedly transitions between an active and inactive throttled
condition.
Moonraker Process Statistic Update
Moonraker will emit the following notification each time it samples its process statistics:
{
"jsonrpc": "2.0",
"method": "notify_proc_stat_update",
"params": [{
"moonraker_stats": {
"time": 1615837812.0894408,
"cpu_usage": 1.99,
"memory": 23636,
"mem_units": "kB"
},
"cpu_temp": 44.008,
"network": {
"lo": {
"rx_bytes": 114555457,
"tx_bytes": 114555457,
"bandwidth": 2911.49
},
"wlan0": {
"rx_bytes": 48773134,
"tx_bytes": 115035939,
"bandwidth": 3458.77
}
},
"system_cpu_usage": {
"cpu": 2.53,
"cpu0": 3.03,
"cpu1": 5.1,
"cpu2": 1.02,
"cpu3": 1
},
"websocket_connections": 2
}]
}
As with the proc_stats request the cpu_temp
field will be set to null
if the host machine does not support retrieving CPU
temperatures at /sys/class/thermal/thermal_zone0
.
History Changed
If the [history]
module is enabled the following notification is sent when
a job is added or finished:
{
"jsonrpc": "2.0",
"method": "notify_history_changed",
"params": [
{
"action": "added",
"job": <job object>
}
]
}
The action
field may be added
or finished
. The job
field contains
an object matches the one returned when requesting
job data.
Authorized User Created
If the [authorization]
module is enabled the following notification is
sent when a new user is created:
{
"jsonrpc": "2.0",
"method": "notify_user_created",
"params": [
{
"username": "<username>"
}
]
}
Authorized User Deleted
If the [authorization]
module is enabled the following notification is
sent when an existing user is deleted.
{
"jsonrpc": "2.0",
"method": "notify_user_deleted",
"params": [
{
"username": "<username>"
}
]
}
Service State Changed
Moonraker monitors the state of systemd services it is authorized to track. When the state of a service changes the following notification is sent:
{
"jsonrpc": "2.0",
"method": "notify_service_state_changed",
"params": [
{
"klipper": {
"active_state": "inactive",
"sub_state": "dead"
}
}
]
}
The example above shows that the klipper
service has changed to inactive
.
Job Queue Changed
Moonraker will send a job_queue_changed
notification when a change is
detected to the queue state or the queue itself:
{
"jsonrpc": "2.0",
"method": "notify_job_queue_changed",
"params": [
{
"action": "state_changed",
"updated_queue": null,
"queue_state": "paused"
}
]
}
The object sent with the notification contains the following fields:
action
: The action taken to the queue which led to the notification. Will be a string set to one of the following values:state_changed
: The queue state has changedjobs_added
: One or more jobs were added to the queuejobs_removed
: One or more jobs were removed from the queuejob_loaded
: A job was popped off the queue and successfull started
updated_queue
: If the queue itself is changed this will be a list containing each item in the queue. If the queue has not changed this will benull
.queue_state
: The current queue state
Button Event
Moonraker [button]
components may be configured to emit websocket
notifications.
{
"jsonrpc": "2.0",
"method": "notify_button_event",
"params": [
{
"name": "my_button",
"type": "gpio",
"event": {
"elapsed_time": 0.09323832602240145,
"received_time": 698614.214597004,
"render_time": 698614.214728513,
"pressed": false
},
"aux": null
}
]
}
The params
array will always contain a single object with the following
fields:
name
: The name of the configured buttontype
: The button type, currently this will always begpio
event
: An object with details about the button event, containing the following fields:elapsed_time
: The time elapsed (in seconds) since the last detected button eventreceived_time
: The time the event was detected according to asyncio's monotonic clock. Note that this is not in "unix time".render_time
: The time the template was rendered (began execution) according to asyncio's montonic clock. It is possible execution of an event may be delayed well beyond thereceived_time
.pressed
: A boolean value to indicate if the button is currently pressed.
aux
: This is an optional field where the button may specify any json encodable value. Clients may suggest a specific button configuration that includes details about the event. If no aux parameter is specified in the configuration this will be anull
value.
Announcement update event
Moonraker will emit the notify_announcement_update
notification when
a announcement entries are addded or removed:
{
"jsonrpc": "2.0",
"method": "notify_announcement_update",
"params": [
{
"entries": [
{
"entry_id": "arksine/moonlight/issue/3",
"url": "https://github.com/Arksine/moonlight/issues/3",
"title": "Test announcement 3",
"description": "Test Description [with a link](https://moonraker.readthedocs.io).",
"priority": "normal",
"date": 1647459219,
"dismissed": false,
"date_dismissed": null,
"dismiss_wake": null,
"source": "moonlight",
"feed": "moonlight"
},
{
"entry_id": "arksine/moonlight/issue/2",
"url": "https://github.com/Arksine/moonlight/issues/2",
"title": "Announcement Test Two",
"description": "This is a high priority announcement. This line is included in the description.",
"priority": "high",
"date": 1646855579,
"dismissed": false,
"date_dismissed": null,
"dismiss_wake": null,
"source": "moonlight",
"feed": "moonlight"
}
{
"entry_id": "arksine/moonraker/issue/349",
"url": "https://github.com/Arksine/moonraker/issues/349",
"title": "PolicyKit warnings; unable to manage services, restart system, or update packages",
"description": "This announcement is an effort to get ahead of a coming change that will certainly result in issues. PR #346 has been merged, and with it are some changes to Moonraker's default behavior.",
"priority": "normal",
"date": 1643392406,
"dismissed": false,
"source": "moonlight",
"feed": "Moonraker"
}
]
}
]
}
The params
array will contain an object with all current announcement entries.
This object is identical to that returned by the
list announcements endpoint.
Announcement dismissed event
Moonraker will emit the notify_announcement_dismissed
notification when
a dismissed announcement is detected:
{
"jsonrpc": "2.0",
"method": "notify_announcement_dismissed",
"params": [
{
"entry_id": "arksine/moonlight/issue/3"
}
]
}
The params
array will contain an object with the entry_id
of the dismissed
announcement.
Announcement wake event
Moonraker will emit the notify_announcement_wake
notification when
a specified wake_time
for a dismissed announcement has expired.
{
"jsonrpc": "2.0",
"method": "notify_announcement_wake",
"params": [
{
"entry_id": "arksine/moonlight/issue/1"
}
]
}
The params
array will contain an object with the entry_id
of the
announcement that is no longer dismissed.
Appendix
Websocket setup
The websocket is located at ws://host:port/websocket
, for example:
var s = new WebSocket("ws://" + location.host + "/websocket");
!!! tip A client using API Key authorization may request a oneshot token, applying the result to the websocket request's query string:
ws://host:port/websocket?token={32 character base32 string}
The following startup sequence is recommened for clients which make use of the websocket:
- Attempt to connect to
/websocket
until successful using a timer-like mechanism - Once connected, query
/server/info
(orserver.info
) for the ready status.- If the response returns an error (such as 404) then either the client
is not authorized or Moonraker is not running. Direct the user to
SSH into the machine and check
/tmp/moonraker.log
. - If the response returns success, check the result's
klippy_state
field:klippy_state == "ready"
: you may proceed to request status of printer objects make subscriptions, get the file list, etc.klippy_state == "error"
: Klippy has experienced an error starting upklippy_state == "shutdown"
: Klippy is in a shutdown state.klippy_state == "startup"
: re-request/server/info
in 2 seconds.- If
error
orshutdown
is detected it might be wise to prompt the user. You can get a description from thestate_message
field of a/printer/info
request.
- If
- If the response returns an error (such as 404) then either the client
is not authorized or Moonraker is not running. Direct the user to
SSH into the machine and check
- Repeat step 2 until Klipper reports ready.
- Clients should watch for the
notify_klippy_disconnected
event. If received then Klippy has either been stopped or restarted. In this state the client should repeat the steps above to determine when klippy is ready.
Basic Print Status
An advanced client will likely use subscriptions and notifications to interact with Moonraker, however simple clients such as home automation software and embedded devices (ie: ESP32) may only wish to monitor the status of a print. Below is a high level walkthrough for receiving print state via polling.
-
Set up a timer to poll at the desired interval. Depending on your use case, 1 to 2 seconds is recommended.
-
On each cycle, issue the following request:
GET http://host/printer/objects/query?webhooks&virtual_sdcard&print_stats
Or via JSON-RPC 2.0:
{ "jsonrpc": "2.0", "method": "printer.objects.query", "params": { "objects": { "webhooks": null, "virtual_sdcard": null, "print_stats": null } }, "id": 5664 }
-
If the request returns an error or the returned
result.status
is an empty object, then this is an indication that Klippy either experienced an error or it is not properly configured. Each queried object should be available inresult.status
. The client should check to make sure that all objects are received before proceeding. -
Inspect
webhooks.ready
. If the value is notready
the printer is not available.webhooks.message
contains a message pertaining to the current state. -
If the printer is ready, inspect
print_stats.state
. It may be one of the following values: -standby
: No print in progress -printing
: The printer is currently printing -paused
: A print in progress has been paused -error
: The print exited with an error.print_stats.message
contains a related error message -complete
: The last print has completed -
If
print_stats.state
is notstandby
thenprint_stats.filename
will report the name of the currently loaded file. -
print_stats.filename
can be used to fetch file metadata. It is only necessary to fetch metadata once per print.GET http://host/server/files/metadata?filename=<filename>
Or via JSON-RPC 2.0:
{ "jsonrpc": "2.0", "method": "server.files.metadata", "params": { "filename": "{filename}" }, "id": 5643 }
If metadata extraction failed then this request will return an error. Some metadata fields are only populated for specific slicers, and unsupported slicers will only return the size and modifed date.
-
There are multiple ways to calculate the ETA, this example will use file progress, as it is possible calculate the ETA with or without metadata.
-
If
metadata.estimated_time
is available, the eta calculation can be done as:// assume "result" is the response from the status query let vsd = result.status.virtual_sdcard; let prog_time = vsd.progress * metadata.estimated_time; let eta = metadata.estimated_time - prog_time
Alternatively, one can simply subtract the print duration from the estimated time:
// assume "result" is the response from the status query let pstats = result.status.print_status; let eta = metadata.estimated_time - pstats.print_duration; if (eta < 0) eta = 0;
-
If no metadata is available, print duration and progress can be used to calculate the ETA:
// assume "result" is the response from the status query let vsd = result.status.virtual_sdcard; let pstats = result.status.print_stats; let total_time = pstats.print_duration / vsd.progress; let eta = total_time - pstats.print_duration;
-
-
It is possible to query additional objects if a client wishes to display more information (ie: temperatures). See the Printer Objects documentation for details.
Bed Mesh Coordinates
The Bed Mesh printer object may be used to generate three dimensional coordinates of a probed area (or mesh). Below is an example (in javascript) of how to transform the data received from a bed_mesh object query into an array of 3D coordinates.
// assume that we have executed an object query for bed_mesh and have the
// result. This example generates 3D coordinates for the probed matrix,
// however it would work with the mesh matrix as well
function process_mesh(result) {
let bed_mesh = result.status.bed_mesh
let matrix = bed_mesh.probed_matrix;
if (!(matrix instanceof Array) || matrix.length < 3 ||
!(matrix[0] instanceof Array) || matrix[0].length < 3)
// make sure that the matrix is valid
return;
let coordinates = [];
let x_distance = (bed_mesh.mesh_max[0] - bed_mesh.mesh_min[0]) /
(matrix[0].length - 1);
let y_distance = (bed_mesh.mesh_max[1] - bed_mesh.mesh_min[1]) /
(matrix.length - 1);
let x_idx = 0;
let y_idx = 0;
for (const x_axis of matrix) {
x_idx = 0;
let y_coord = bed_mesh.mesh_min[1] + (y_idx * y_distance);
for (const z_coord of x_axis) {
let x_coord = bed_mesh.mesh_min[0] + (x_idx * x_distance);
x_idx++;
coordinates.push([x_coord, y_coord, z_coord]);
}
y_idx++;
}
}
// Use the array of coordinates visualize the probed area
// or mesh..
Converting to Unix Time
Some of Moonraker's APIs return a date represented in Unix time. Most languanges have functionality built in to convert Unix time to a workable object or string. For example, in JavaScript one might do something like the following:
for (let resp of result.gcode_store) {
let date = new Date(resp.time * 1000);
// Do something with date and resp.message ...
}
Announcements
Moonraker announcements are effectively push notifications that can be used to notify users of important information related the development and status of software in the Klipper ecosystem. This section will provide an overview of how the announcement system works, how to set up a dev environment, and provide recommendations on client implementation.
How announcements work
Moonraker announcements are GitHub issues tagged with the announcement
label. GitHub repos may registered with
moonlight, which is responsible
for generating RSS feeds from GitHub issues using GitHub's REST API. These
RSS feeds are hosted on GitHub Pages, for example Moonraker's feed may be found
here. By
centralizing GitHub API queries in moonlight
we are able to poll multiple
repos without running into API rate limit issues. Moonlight has has a workflow
that checks all registered repos for new announcements every 30 minutes. In
theory it would be able to check for announcements in up to 500 repos before
exceeding GitHub's API rate limit.
Moonraker's [announcements]
component will always check the klipper
and
moonraker
RSS feeds. It is possible to configure additional RSS feeds by
adding them to the subscriptions
option. The component will poll configured
feeds every 30 minutes, resulting in maximum of 1 hour for new announcements
to reach all users.
When new issues are tagged with announcement
these entries will be parsed
and added to the RSS feeds. When the issue is closed they will be removed from
the corresponding feed. Moonlight will fetch up to 20 announcements for each
feed, if a repo goes over this limit older announcements will be removed.
!!! Note It is also possible for Moonraker to generate announcements itself. For example, if a Moonraker component needs user feedback it may generate an announcement and notify all connected clients. From a client's perspective there is no need to treat announcements differently than any other announcement.
Setting up the dev environment
Moonraker provides configuration to parse announcements from a local folder so that it is possible to manually add and remove entries, allowing client developers to perform integration tests:
# moonraker.conf
[announcements]
dev_mode: True
With dev_mode
enabled, Moonraker will look formoonraker.xml
and
klipper.xml
in the following folder:
~/moonraker/.devel/announcement_xml
If moonraker is not installed in the home folder then substitute ~
for the parent folder location. This folder is in a hardcoded location
to so as not to expose users to vulnerabilites associated with parsing XML.
It is possible to configure Moonraker to search for your own feeds:
# moonraker.conf
[announcements]
subscription:
my_project
dev_mode: True
The above configuration would look for my_project.xml
in addition to
klipper.xml
and moonraker.xml
. The developer may manually create
the xml feeds or they may clone moonlight
and leverage its script
to generate a feed from issues created on their test repo. When local
feeds have been modified one may call the update announcements API to have Moonraker fetch the updates and add/remove
entries.
RSS file structure
Moonlight generates RSS feeds in XML format. Below is an example generated from moonlight's own issue tracker:
<?xml version='1.0' encoding='utf-8'?>
<rss version="2.0" xmlns:moonlight="https://arksine.github.io/moonlight">
<channel>
<title>arksine/moonlight</title>
<link>https://github.com/Arksine/moonlight</link>
<description>RSS Announcements for Moonraker</description>
<pubDate>Tue, 22 Mar 2022 23:19:04 GMT</pubDate>
<moonlight:configHash>f2912192bf0d09cf18d8b8af22b2d3501627043e5afa3ebff0e45e4794937901</moonlight:configHash>
<item>
<title>Test annoucement 3</title>
<link>https://github.com/Arksine/moonlight/issues/3</link>
<description>Test Description [with a link](https://moonraker.readthedocs.io).</description>
<pubDate>Wed, 16 Mar 2022 19:33:39 GMT</pubDate>
<category>normal</category>
<guid>arksine/moonlight/issue/3</guid>
</item>
<item>
<title>Announcement Test Two</title>
<link>https://github.com/Arksine/moonlight/issues/2</link>
<description>This is a high priority announcement. This line is included in the description.</description>
<pubDate>Wed, 09 Mar 2022 19:52:59 GMT</pubDate>
<category>high</category>
<guid>arksine/moonlight/issue/2</guid>
</item>
<item>
<title>Announcement Test One</title>
<link>https://github.com/Arksine/moonlight/issues/1</link>
<description>This is the description. Anything here should appear in the announcement, up to 512 characters.</description>
<pubDate>Wed, 09 Mar 2022 19:37:58 GMT</pubDate>
<category>normal</category>
<guid>arksine/moonlight/issue/1</guid>
</item>
</channel>
</rss>
Each xml file may contain only one <rss>
element, and each <rss>
element
may contain only one channel. All items must be present aside from
moonlight:configHash
, which is used by the workflow to detect changes to
moonlight's configuration. Most elements are self explanatory, developers will
be most interested in adding and removing <item>
elements, as these are
the basis for entries in Moonraker's announcement database.
Generating announcements from your own repo
As mentioned previously, its possible to clone moonlight and use its rss script to generate announcements from issues in your repo:
cd ~
git clone https://github.com/arksine/moonlight
cd moonlight
virtualenv -p /usr/bin/python3 .venv
source .venv/bin/activate
pip install httpx[http2]
deactivate
To add your repo edit ~/moonlight/src/config.json
:
{
"moonraker": {
"repo_owner": "Arksine",
"repo_name": "moonraker",
"description": "API Host For Klipper",
"authorized_creators": ["Arksine"]
},
"klipper": {
"repo_owner": "Klipper3d",
"repo_name": "klipper",
"description": "A 3D Printer Firmware",
"authorized_creators": ["KevinOConnor"]
},
// Add your test repo info here. It should contain
// fields matching those in "moonraker" and "klipper"
// shown above.
}
Once your repo is added, create one or more issues on your GitHub
repo tagged with the announcement
label. Add the critical
label to
one if you wish to test high priority announcements. You may need to
create these labels in your repo before they can be added.
Now we can use moonlight to generate the xml files:
cd ~/moonlight
source .venv/bin/activate
src/update_rss.py
deactivate
After the script has run it will generate the configured RSS feeds
and store them in ~/moonlight/assets
. If using this method it may
be useful to create a symbolic link to it in Moonraker's devel folder:
cd ~/moonraker
mkdir .devel
cd .devel
ln -s ~/moonlight/assets announcement_xml
If you haven't done so, configure Moonraker to subscribe to your feed and restart the Moonraker service. Otherwise you may call the announcement update API to have Moonraker parse the announcements from your test feed.
Implementation details and recommendations
When Moonraker detects a change to one or more feeds it will fire the announcement update notification. It is also possible to query the API for announcements. Both the notification and the API return a list of announcement entries, where each entry is an object containing the following fields:
entry_id
: A unique ID derived for each entry. Typically this is in the form of{owner}/{repo}/issue/{issue number}
.url
: The url to the full announcement. This is generally a link to an issue on GitHub.title
: Announcement title, will match the title of the issue on GitHub.description
: The first paragraph of the announcement. Anything over 512 characters will be truncated.priority
: Can benormal
orhigh
. It is recommended that clients immediately alert the user when one or morehigh
priority announcments are present. Issued tagged with thecritical
label will be assigned ahigh
priority.date
: The announcement creation date in unix time.dismissed
: If set totrue
this announcement has been previously dismisseddate_dismissed
: The date the announcement was dismissed in unix time. If the announcement has not been dismissed this value isnull
.dismiss_wake
: If the announcement was dismissed with awake_time
specified this is the time (in unix time) at which thedismissed
state will revert. If the announcement is not dismissed or dismissed indefinitely this value will benull
.source
: The source from which the announcement was generated. Can bemoonlight
orinternal
.feed
: The RSS feed for moonlight announcements. For example, this could beMoonraker
orKlipper
. If the announcement was generated internally this should match the name of the component that generated the announcement.
When a client first connects to Moonraker it is recommended that the list announcements API is called to retreive the current list of entries. A client may then watch for the announcement update and announcement dismissed notifications and update the UI accordingly.
Client devs should decide how they want to present announcements to users. They could be treated as any other notification, for example a client
may have a notification icon that shows the current number of unread
announcements. Clients can mark an announcement as read
by calling
the dismiss announcement API. Any announcement
entry with dismissed = true
should be considered read.
When a high priority
announcement is detected it is recommended that
clients present the announcement in a format that is immediately visible
to the user. That said, it may be wise to allow users to opt out of
this behavior via configuration.
!!! Note
If an announcement is dismissed, closed, then reopened the
dismissed
flag will reset to false. This is expected behavior
as announcements are pruned from the database when they are no
longer present in feeds. It isn't valid for repo maintaners
to re-open a closed announcement. That said, its fine to close
and re-open issues during development and testing using repos
that are not yet registered with moonlight.