Introduction
figmaid is a simple web server that serves locally installed fonts to Figma.
How does it work? (in a nutshell)
Figma allows local fonts through a background service they call FigmaAgent. It works by establishing a web server with two jobs: serving locally installed fonts and a single index file with metadata about these fonts.
It comes with some limitations though, most importantly the platform support, or lack thereof. FigmaAgent currently only supports Windows and macOS. figmaid runs on Linux natively and can run on Windows and macOS via Docker.
Here's what Figma does when you load a document:
- Request index file from localhost (that's your computer!)
- Add fonts from the JSON response to selection
- If user uses one of these fonts get the file
figmaid is a workhorse. It eats fonts and spits out their metadata simply by walking through your directories (platform defaults, and any you specify in configuration), parses font tables and creates the index file.
In practice
- Start the server and reload Figma, if open. Figma will now use fonts served by figmaid.
- Individual font files are requested from the server.
Install
figmaid currently only supports Linux. Previous versions included precompiled binaries for macOS (arm64 and x64) as well as Windows. For the documentation of the last macOS/Windows version switch to tag v0.9.0. If you want to use custom directories and thus need figmaid on macOS/Windows, you can still use the Docker image and continue to get updates.
There's four ways you can install figmaid
-
With the installation script on Linux
curl -fsSL https://raw.githubusercontent.com/jqpe/figmaid/main/install.sh | sh
-
Downloading a precompiled binary from releases
-
Using the Docker image, documented in figmaid in docker
-
Building from source
You can use figmaid on Windows/macOS via Docker, but you're probably better off using FigmaAgent unless you need custom directories. If you only intend on using installed fonts, FigmaAgent will pick those up.
To build from source install Rust. For a specific tag you can use the --tag
argument, e.g. for Windows and macOS you're limited to <= v0.9.0.
cargo install --git https://github.com/jqpe/figmaid
Configuration
The default configuration looks like below:
{
"$schema": "https://raw.githubusercontent.com/jqpe/figmaid/main/docs/schema.json",
"port": 18412,
"directories": [
"/usr/share/fonts",
"/usr/local/share/fonts",
"~/.local/share/fonts"
]
}
The configuration file lives in $HOME/.config/figmaid/figmaid.json. Additionally you can use environment variables PORT and DIRS, they will have preference.
If you are using Ubuntu or Fedora no additional configuration is required if you only wish to use installed fonts. Other Linux distros may or may not have the same directories, see your font configuration file to be sure.
Custom directories figmaid doesn't care about where your font files are located or whether they're installed or not. To add a custom directory, specify it in
directories
. The path to the directory doesn't need to be absolute, if you have fonts in /Design/fonts/<sub_folder>/**/* you can just specify /Design/fonts.
Note about directories
- The directory path doesn't have to be absolute, but shortcuts like the tilde ~ are not supported.
- figmaid has to have access to the directory. This usually means that the user that started figmaid has to have access.
- On a related note: if you run as sudo, the home directory which figmaid uses to locate your configuration file will not be what you expect. If you for some reason want to run figmaid as sudo run the command with
HOME=/home/<user> figmaid
- On a related note: if you run as sudo, the home directory which figmaid uses to locate your configuration file will not be what you expect. If you for some reason want to run figmaid as sudo run the command with
Command-line interface
figmaid has a small CLI to perform common tasks. It ships as a single executable with a single subcommand to work with the configuration.
figmaid
Starts the server. It starts on the foreground by default.
figmaid config
Create, open and validate configuration
Run without subcommands to print current directories and amount of loaded fonts
USAGE:
figmaid config <SUBCOMMAND>
OPTIONS:
-h, --help Print help information
SUBCOMMANDS:
create Create default configuration file
help Print this message or the help of the given subcommand(s)
open Open configuration file in text editor
validate Validate configuration
Usage
This section contains guides for different use cases like running figmaid in the background.
Running figmaid in the background
figmaid doesn't run in the background by default. You can configure this yourself if you need it.
Docker
Docker has a powerful feature called restart policies, which allows containers to start automatically if they exit or if Docker restarts, as such, running figmaid inside Docker is a great way of keeping figmaid alive. Read more in the dedicated Docker section: figmaid in Docker.
As systemd unit
If you are using Ubuntu, or a similar distro that uses systemd, you can create a unit file for figmaid in just a few minutes.
Prerequisites:
- You are a sudoer
- You know the executable path of figmaid (use
which figmaid
) - You know your username
- Ideally you'd have a basic understanding of what systemd units do. Read more at the official systemd website.
While systemd supports user units, having them reload on reboot is not as straightforward. It would require enabling lingering with loginctl
, with the major caveat that every single user service would be started on reboot.
Create the unit file
- Below, replace
<your username>
with the user you installed figmaid to.- If you used the installation script, the executable path is
/home/<your username>/.local/bin/figmaid
.- Replace
<figmaid executable>
with the path to figmaid.
The code block is editable, and changes you made will be present if you click the copy button.
Change the User
and ExecStart
properties as described above.
[Unit]
Description=figmaid — font daemon for figma
Documentation=https://figmaid.pages.dev
Wants=network-online.target
After=network-online.target
[Service]
User=<your username>
ExecReload=/bin/kill -HUP $MAINPID
ExecStart=<figmaid executable>
Restart=on-failure
RestartSec=2
[Install]
WantedBy=multi-user.target
Save the file to /etc/systemd/system/figmaid.service
.
Then, to enable, start, and view the status of the service run the below commands:
sudo systemctl enable figmaid
sudo systemctl start figmaid
sudo systemctl status figmaid
View logs
sudo journalctl -eu figmaid
Here -e
tells journalctl that we want it to automatically scroll to the end. -u
tells that we're asking for an unit.
Update configuration
Currently figmaid doesn't reload the configuration when you reload the service. Instead, you must restart the service.
sudo systemctl restart figmaid
Note that for already configured directories, no restarting is necessary. Instead, you simply have to add the font to the configured directory and reload Figma.
figmaid in Docker
You can also run figmaid inside a container. This comes with obvious benefits such as Docker's restart policies, which let you delegate restarting figmaid to Docker so you don't have to. The minor caveat is that specifying additional directories is more of a chore and the container includes an operating system, adding roughly 50MB.
Nevertheless, the benefits overshadow the caveats here for most people. Let's see it!
docker run -d -p 18412:18412 \
-v /usr/share/fonts:/usr/share/fonts:ro \
-v ~/.config/figmaid/figmaid.json:/root/.config/figmaid/figmaid.json \
--name figmaid --restart always \
nykanen/figmaid
Step by step — what is the above telling Docker? First, the -d
short flag starts the container in detached mode, meaning there's no shell output nor input. Secondly, you must bind the port 18412 to your local computer, this is done with the -p
short flag. Otherwise figmaid will only be accessible from within the container which is not very useful.
The -v
short flag stands for volume, and as you'd expect, it binds a directory from your computer to be accessible within the container. Again, this is a requirement in almost all cases, unless you intend to install fonts inside the container, which by any means you could do. Above, we expose two directories: the directory for installed fonts in Ubuntu / Fedora, and the configuration file. The short :ro
option enforces that figmaid
doesn't have access to write to those files — completely optional as figmaid only reads those files, but explicitness is great!
That's the gist of it, but take the time to understand the arguments. Before running the command blindly it is probably beneficial to lay out a plan so figmaid works for you in the future with minimal hassle. These docs will still be waiting for you if you need to change stuff later.
After succesfully configuring and starting figmaid, the output of docker container ls
would be similar to below:
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS
c07e070bb470 nykanen/figmaid "sh -c 'HOST=${HOST}…" 0 days ago Up 0 hours 0.0.0.0:18412->18412/tcp, :::18412->18412/tcp
Take note of the PORTS value. 0.0.0.0:18412 is available from 18412/tcp on your local machine. To make sure that everything really went as expected, open http://localhost:18412/figma/font-files on a browser and you should get a big chunk of JSON.
Adding additional directories
Simply changing your configuration file is not enough, you must bind the directory from your host to the container. If you find you're adding a lot of directories, you can use figmaid's ability to walk directories. If you have fonts in $HOME/design/fonts and $HOME/Downloads specifying just $HOME might be the way to go. This is not free though — figmaid would walk all directories in $HOME — so pick a good middleground, unfortunately this is up to you to figure out.
To add new bind mounts to figmaid you must first delete the first iteration, e.g. docker rm -f figmaid
then create a new container with the additional bind mount:
docker run -d -p 18412:18412 \
-v /usr/share/fonts:/usr/share/fonts:ro \
-v ~/Downloads:/root/Downloads:ro \
-v ~/.config/figmaid/figmaid.json:/root/.config/figmaid/figmaid.json:rw \
--name figmaid --restart always \
nykanen/figmaid
Above we bound the default directory for downloaded files on Ubuntu to /root/Downloads
. Remember to add this directory to your configuration file by editing ~/.config/figmaid/figmaid.json
and add the directory for what the directory path is inside the container. In this case our configuration file would contain the following directories: /usr/share/fonts
and /root/Downloads
.
Verify that directories are included
After adding directories it would be nice to know they actually are picked up by figmaid. Fortunately this is not that big of a checklist, and often you would do this only once, never having to think about it again.
To get a list of all mounts inside a container use
docker inspect figmaid --format "{{range .Mounts}}{{ .Source }} -> {{ .Destination }}
{{ end }}"
You should get an output similar to below:
/usr/share/fonts -> /usr/share/fonts
/home/$user/.config/figmaid/figmaid.json -> /root/.config/figmaid/figmaid.json
The left-hand side is the directory on your host machine. This would contain your configuration file and font directories. The right-hand side is what the path is inside the container. Therefore it's also the path you should have in your configuration file.
Let's verify this next. Open your configuration file on your host machine (whatever is on the left-hand side of -> /root/.config/figmaid/figmaid.json.) As introduced in configuration, the directories property lets you specify additional directories. In the above example we want to include the right-hand side of /usr/share/fonts, which just happens to be /usr/share/fonts, but it could be anything! After ensuring that you're using the correct directories, and checking that the configuration is valid (docker exec -it figmaid figmaid config validate
), all there is left to do is refresh Figma, if open. You can also run docker exec -it figmaid figmaid config
to list loaded directories.
Troubleshooting
When things don't go as expected this section contains instructions for common pitfalls.
My fonts are not showing up
Check that you have the server running: sudo netstat -l | grep 18412
, or with ss: sudo ss -l | grep 18412
If it prints nothing, start the server with figmaid
. At this point if figmaid
panics with AddrInUse
, see relevant docs.
If it prints something, but you are still not seeing your fonts:
- Ensure you have the font installed or you have specified the directory where it lives in, see usage.
- Reload Figma. Figma requests fonts when it's initialized and adding new fonts requires a refresh.
- Run
figmaid config validate
to check that your configuration file is valid. If you accidentally mispelled something figmaid will fallback to the default configuration.- For a quick fix run DIRS=/your/directory figmaid
- Run
figmaid config open
and check that you specified the correct directory. - Run
figmaid config
, this lists all directories and amount of loaded font files per directory. If you want to be extra spicy you can specify an absolute path to the direct parent of your font file withDIRS=/my/fontdir figmaid config
. If figmaid picked up some fonts from this directory it would print your directory path followed by amount of fonts greater than zero. - If all else fails, the font might not have correct tables. I'd apprecatiate if you'd take the time to open a issue with the font file included.
figmaid figures out the font family by looking at it's metadata. In some cases the font family and filename don't match — in this case you should install the font and then open it in your platform's GUI and see what the font family actually is.
AddrInUse
This error happens when figmaid is trying to bind a port that is already in use by another program.
Solution
-
See what program is using the port
sudo netstat -ltnp | grep '18412'
- If you don't have
netstat
you can replace it withss
, the output will be a little different.
-
Kill the program
0.0.0.0:* LISTEN 70793/figmaid ^pid ^process name
- With process name:
pkill -e figmaid
- With process ID:
kill 70793
- With process name:
For services you can forcibly stop the service by running above commands as sudo, but
stopping the service with sudo systemctl stop <process name>
is recommended.