Metadata-Version: 2.4
Name: spoolman2slicer
Version: 0.10.0
Summary: Export active Spoolman spools' data to filament configuration files for 3D printer slicers
Author-email: Sebastian Andersson <sebastian@bittr.nu>
License-Expression: GPL-3.0-or-later
Project-URL: Homepage, https://github.com/bofh69/spoolman2slicer
Project-URL: Repository, https://github.com/bofh69/spoolman2slicer
Project-URL: Issues, https://github.com/bofh69/spoolman2slicer/issues
Keywords: spoolman,3d-printing,slicer,orcaslicer,prusaslicer,superslicer
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: End Users/Desktop
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Programming Language :: Python :: 3.14
Classifier: Topic :: Printing
Requires-Python: >=3.9
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: appdirs==1.4.4
Requires-Dist: Jinja2==3.1.6
Requires-Dist: requests==2.32.4
Requires-Dist: websockets==12.0
Dynamic: license-file

<!--
SPDX-FileCopyrightText: 2024-2025 Sebastian Andersson <sebastian@bittr.nu>

SPDX-License-Identifier: GPL-3.0-or-later
-->

[![REUSE status](https://api.reuse.software/badge/github.com/bofh69/spoolman2slicer)](https://api.reuse.software/info/github.com/bofh69/spoolman2slicer)
![GitHub Workflow Status](https://github.com/bofh69/spoolman2slicer/actions/workflows/pylint.yml/badge.svg)

# Spoolman to slicer config generator

## Intro

*spoolman2slicer* is a program to export Spoolman's data to filament
configuration files for:

* [OrcaSlicer](https://github.com/SoftFever/OrcaSlicer)
* [PrusaSlicer](https://www.prusa3d.com/page/prusaslicer_424/)
* [SuperSlicer](https://github.com/supermerill/SuperSlicer)

The filament configuration files are created from templates.

## Table of Contents

<!--TOC-->

- [Spoolman to slicer config generator](#spoolman-to-slicer-config-generator)
  - [Intro](#intro)
  - [The workflow](#the-workflow)
    - [Files from filaments](#files-from-filaments)
    - [Files from spools](#files-from-spools)
  - [Usage](#usage)
  - [Installation](#installation)
    - [From PyPI (Recommended)](#from-pypi-recommended)
    - [Using Docker/Docker-Compose](#using-dockerdocker-compose)
    - [From Source](#from-source)
  - [Configuring the filament config templates](#configuring-the-filament-config-templates)
    - [Intro](#intro-1)
    - [Where the files are read from](#where-the-files-are-read-from)
    - [Available variables in the templates](#available-variables-in-the-templates)
    - [Creating templates from existing config](#creating-templates-from-existing-config)
    - [Writing the templates](#writing-the-templates)
  - [Generating for multiple printers, the variants argument](#generating-for-multiple-printers-the-variants-argument)
  - [Running examples](#running-examples)
    - [Ubuntu & OrcaSlicer](#ubuntu--orcaslicer)
    - [Ubuntu & PrusaSlicer](#ubuntu--prusaslicer)
    - [Ubuntu & SuperSlicer](#ubuntu--superslicer)
    - [MacOs & OrcaSlicer](#macos--orcaslicer)
  - [Contributing](#contributing)

<!--TOC-->

## The workflow

### Files from filaments

You add your spools' manufacturers, filaments and the spools to Spoolman.

For each filament that has at least one active spool, spoolman2slicer creates a
filament configuration based on the filament.

The next time you start the slicer you will see the available filaments.

The templates contain a "filament_start_gcode" field, `ASSERT_ACTIVE_FILAMENT ID={{id}}`,
which comes from
[this file](https://github.com/bofh69/nfc2klipper/blob/v0.0.4/klipper-spoolman.cfg)
in my other repo, [nfc2klipper](https://github.com/bofh69/nfc2klipper).

That macro checks that the `active_filament` variable is the same as
the choosen filament.

The `active_filament` variable is set by first calling `SET_ACTIVE_FILAMENT ID=`
and the id. That is called automatically by `nfc2klipper`.

If not using `nfc2klipper`, there is a Moonraker agent
[spool2klipper](https://github.com/bofh69/spool2klipper)
that can be used to update the active_filament variable whenever
the spool is changed in moonraker (via frontends, code macros) etc.


### Files from spools

A different workflow is possible and supported by spoolman2slicer.

Instead of letting the slicer's generated gcode to verify that the right
filament has been loaded (no matter which spool it comes from), one
can make it set the spool in Klipper during the start of the print.

The `--create-per-spool` command line option causes spoolman2slicer
to generate one filament configuration file per spool, one for the most
used spool or one for the latest used spool.  The default templates will then use
"SET_ACTIVE_SPOOL ID={{spool.id}}" in the "filament_start_gcode" field.

When creating one file for every spool, it uses the `filename_for_spool.template`
file to create the filenames, otherwise the `filename.template` file
is used to create the filenames.

The included `filename_for_spool.template` simply appends the spool's id
to the filename. That's often not very helpful for selecting the right
spool in the slicer. Depending on the data you've added to the spool's
different fields could be better. The `{{spool}}` field contains all
the fields from the Spoolman's spool object.
`{{spool.lot_nr}}` or `{{spool.location}}` might be better options for
you. You can also add an extra field in Spoolman's config for the spools
and use that field here, ie `{{spool.extra.my_label}}`.

The default template files also use `- {{spool.id}}` at
the end of the "name" field in the templates that use a name.
If you update the filename template, update the name field too.


## Usage

```text
usage: spoolman2slicer.py [-h] [--version] -d DIR
                          [-s {orcaslicer,prusaslicer,slic3r,superslicer}]
                          [-u URL] [-U] [-v] [-V VALUE1,VALUE2..] [-D]
                          [--create-per-spool {all,least-left,most-recent}]

Fetches data from Spoolman and creates slicer filament config files.

options:
  -h, --help            show this help message and exit
  --version             show program's version number and exit
  -d DIR, --dir DIR     the slicer's filament config dir
  -s {orcaslicer,prusaslicer,slic3r,superslicer}, --slicer {orcaslicer,prusaslicer,slic3r,superslicer}
                        the slicer
  -u URL, --url URL     URL for the Spoolman installation
  -U, --updates         keep running and update filament configs if they're
                        updated in Spoolman
  -v, --verbose         verbose output
  -V VALUE1,VALUE2.., --variants VALUE1,VALUE2..
                        write one template per value, separated by comma
  -D, --delete-all      delete all filament configs before adding existing
                        ones
  --create-per-spool {all,least-left,most-recent}
                        create one output file per spool instead of per filament.
                        'all': one file per spool.
                        'least-left': one file per filament for the spool having the least filament left.
                        'most-recent': one file per filament for the spool being most recently used.
```

## Installation

### From PyPI (Recommended)

The easiest way to install spoolman2slicer is from PyPI:

```sh
python3 -m venv path-to-venv
source path-to-venv/bin/activate
pip install spoolman2slicer
```

The default templates files are then under `venv/share/spoolman2slicer`.

`spoolman2slicer` is then runable from the virtual environment:
```sh
source path-to-venv/bin/active
spoolman2slicer
```
or
```sh
path-to-venv/bin/spoolman2slicer
```


### Using Docker/Docker-Compose

spoolman2slicer can be run from docker.

Included is a Dockerfile and docker-compose config. Before using it,
update the environment variables and mount points in docker-compose and
run:
```sh
  docker-compose up -d
```

for other arguments to the command, use:
```
  entrypoint: [ "sh", "-c", "python3 ./spoolman2slicer.py #AddYourArgumentsHere" ]
```


### From Source

If you want to run from source, clone the repository and run:
```sh
python3 -m venv venv
source venv/bin/activate
pip install -r requirements.txt
```

## Configuring the filament config templates

### Intro

spoolman2slicer uses [Jinja2](https://palletsprojects.com/p/jinja/)
templates for the configuration files it creates and it also uses
such a template for the configuration files' names.

### Where the files are read from

The templates are read with the filaments' material's name from
`<configdir>/spoolman2slicer/templates-<slicer>/<material>.<suffix>.template`.

`<configdir>` depends on the operating system:

| OS      | Path                                |
| :------ | :---------------------------------- |
| Linux   | `$HOME/.config`                     |
| MacOS   | `$HOME/Library/Application Support` |
| Windows | `%APPDATA%`                         |

`slicer` is the used slicer (superslicer or orcaslicer).
`<material>` is the material used in the filament in Spoolman, ie PLA, ABS etc.
`<suffix>` is `ini` for Super Slicer, `info` and `json` for Orca Slicer
(it uses two files per filament).

If the material isn't found, "default" is used as the material name.


### Available variables in the templates

The variables available to use in the templates comes from the return
data from Spoolman's filament request, described
[here](https://donkie.github.io/Spoolman/#tag/filament/operation/Get_filament_filament__filament_id__get).

When using the `--create-per-spool` argument, the `spool` field
contains Spoolman's spool's fields, as described
[here](https://donkie.github.io/Spoolman/#tag/spool/operation/Find_spool_spool_get).

spoolman2slicer also adds its own fields under the `sm2s` field:
* name - the name of the tool's program file.
* version - the version of the tool.
* now - the time when the file is created.
* now_int - the time when the file is created as the number of seconds since UNIX' epoch.
* slicer_suffix - the filename's suffix.
* variant - one of the comma separated values given to the `--variants` argument, or an empty string.
* spoolman_url - the URL to spoolman.


The available variables, and their values, can be printed by spoolman2slicer when
the filament is about to be written. Use the `-v` argument as argument
to spoolman2slicer when it is started.

Output can look like this (after pretty printing it):
```python
{
  'id': 17,
  'registered': '2024-10-08T12:23:04Z',
  'name': 'Gilford PLA+ Black',
  'vendor': {
    'id': 8,
    'registered': '2024-10-08T12:20:15Z',
    'name': 'Gilford',
    'extra': {}
  },
  'material': 'PLA',
  'price': 250.0,
  'density': 1.24,
  'diameter': 1.75,
  'weight': 1000.0,
  'spool_weight': 116.0,
  'article_number': '102001A',
  'settings_extruder_temp': 190,
  'settings_bed_temp': 60,
  'color_hex': '000000',
  'extra': {
    'pressure_advance': '0.045'
  },
  'sm2s': {
    'name': 'spoolman2slicer.py',
    'version': '0.0.1',
    'now': 'Sun Jan 26 10:57:51 2025',
    'now_int': 1737885471,
    'slicer_suffix': 'ini',
    'variants': 'printer1',
    'spoolman_url': 'http://localhost:7912'
  }
}
```

### Creating templates from existing config

The `create_template_files.py` program can create basic template files
by copying existing filament config files.

Run it like this:
```sh
./create_template_files.py -s orcaslicer -v
```

If the program doesn't find the slicers' config dir, use the -d option:
```sh
./create_template_files.py -s orcaslicer -v -d "path/to/slicers/filament/config/dir"
```

If that's needed, please create/update a github issue with into about
your operating system, the slicer and the path where you found
the filament config files.

---

If needed, the program will create the templates' config dir and
copy the `filename*.template` files there.

For every filament config file in the slicer, it will create a template
file for its material, unless the material already has a template file.

The generated files might work for you, but you should check them
before using them. The start gcode probably needs updates as well as
the first layer temperatures.


### Writing the templates

The default templates as well as the generated template files were based
on my settings. They assume there is an extra
Spoolman filament field defined called "pressure_advance" and sets the
pressure advance settings based on it. The Orca Slicer files also assumes
one has added the Voron filaments in Orca Slicer as they inherit from them (but that's not
the case for the generated files).

When making your own, it is best to start with the generated files and
update the files' fields to fit your preferences.

In the templates, variables are surrounded by `{{` and `}}`.
For variables with values that contain more variables, you write all
the variable names with a dot between. Ie the vendor's name (`Gilford`
above) is written as: `{{vendor.name}}`. Be careful to use the same style as
the original file. If the file wrote `"Gilford"`, remember to keep the
`"` characters around the variable.

There are two special template files, the `filename.template` and
the `filename_for_spool.template`. They are used to create the name of
the generated files. The default ones work, but if creating config files
per spool, you probably want the spool's lot-nr or location in the name
instead of the default, which is the id number of the spool.

You should also have a `default.<suffix>.template` file (or two with
OrcaSlicer, one for the .info file as well). It/they will be used if there is no
material specific template file. The default file is just a copy of
the default PLA template.


The template's format supports advanced features. Follow the link above
to jinja2 to read its documentation.

The [Jinja Playground](https://github.com/bofh69/jinja-playground) program can help
when editing the templates, by providing instant feedback.


## Generating for multiple printers, the variants argument

When using the `--variants` argument, it should have two or more more
values, separated by commas. Ie `--variants printer_small,printer_big`.

spoolman2slicer then generates one set of files per given value.
Each time `sm2s.variant` will have one of the values.

The templates can check the variable and output different fields or values depending on it.
Ie:
```
start_filament_gcode = "; Filament gcode\nSET_PRESSURE_ADVANCE={% if sm2s.variant == "printer_big %}{{extra.pressure_advance_big|default(0)|float}}{% else %}"{{extra.pressure_advance_small|default(0)|float}}{% endif %}\n"
```

The default `filename*.template` files uses the variant variable to put
the variant first in the filename, if given. The other template files don't use it.


## Running examples

When installed from pip, remove use `spoolman2slicer` instead of `./spoolman2slicer.py`.

### Ubuntu & OrcaSlicer
```sh
./spoolman2slicer.py -s orcaslicer -U -d ~/.config/OrcaSlicer/user/default/filament/
```

### Ubuntu & PrusaSlicer
```sh
./spoolman2slicer.py -s prusaslicer -U -d ~/.var/app/com.prusa3d.PrusaSlicer/config/PrusaSlicer/filament/
```

### Ubuntu & SuperSlicer
```sh
./spoolman2slicer.py -U -d ~/.config/SuperSlicer/filament/
```

### MacOs & OrcaSlicer
```sh
./spoolman2slicer.py -s orcaslicer -U -d  ~/Library/Application\ Support/OrcaSlicer/user/default/filament
```

See the other options above.

## Contributing

Interested in contributing to spoolman2slicer? Please check out our [CONTRIBUTING.md](CONTRIBUTING.md) file for guidelines on how to contribute to the project.
