Livecoding for 3D printers
Experiments in live computational sculpting
LivePrinter is an open source system for live, intuitive, immediate drawing and fabrication with 3D printers. It allows people to directly draw shapes, lines, and complex objects using only code. The system serves as an introduction to the basics of 3D printing, as well as platform for experimenting with new techniques that are not allowed by current software. It also allows users to create physical forms using different functions that take into account physical properties like speed and temperature, instead of the usual visual modelling techniques. As a live performance system, LivePrinter blends mechanical music, sculpture and moving image together. It allows musical concepts and visual aesthetics to directly influence digital manufacturing toolpaths and vice versa. It demonstrates a possible future path where humans play a direct, creatively-rewarding part in a system of semi-automated, digital manufacturing.
There’s a forum on https://talk.lurk.org/channel/liveprinter
You can join our email list to keep up to date with performances, demonstrations, workshops, and important updates.
Evan is on Twitter at @evanraskob and can answer questions, etc.
We can create a Reddit forum or YouTube channel if there is demand for it.
A quickstart document can be found here — this needs an update! It is fairly out of date.
The server front-end is coded in JavaScript using NodeJS’s npm build system. You will need to compile it first (or download a pre-compiled release) so download and unzip the project using the link in the above right. Then, in your project folder, open a command prompt. Make sure you have npm installed.
npm install
to install required modulessudo npm install node-gyp -g
before running npm install.
sudo usermod -a -G tty $USERNAME
and sudo usermod -a -G dialout $USERNAME
first before running the server (you might need to reboot your machine after this to see the changes). Then assuming you are on /dev/ttyACM0
try running echo "G28" > /dev/ttyACM0
in your terminal once connected via USB (check for your serial port using dmesg
) to run a standard bed leveling GCode and see if anything happens. Then, unplug and re-plug your printer usb and start the server. npm run watch
(for some reason, build doesn’t always work but watch does) and following the instructions below to get the python server up and running.npm run watch
to have any saved changed automatically re-compiled.npm run server
to run the server, orThe web server (tornado 6) runs on Python 3 (3.6, 3.7 and 3.8+ tested) so you will need that installed. Then, use pip (the Python package manager) to install supporting libraries python3 -m pip install pyserial tornado tornado-jsonrpc2
:
Run the server using your Python environment of choice. The server script is in the liveprinter
subfolder and is called LivePrinterServer.py
, so change to that folder using cd liveprinter
, make sure there’s a logs
folder (create it, or on Linux/OS X in terminal type mkdir logs
) and run:
python LivePrinterServer.py
By default, the server runs on port 8888 so open a web browser to http://localhost:8888 (or change it). It can be run without an actual 3D printer using a “fake” serial port, check the videos or look for the dummy printer in the drop down menu when you’ve got it running.
Here is a short (old!) video demonstrating the python installation on Windows (on OS X it’s the same, just use Terminal instead of Command Prompt):
https://youtu.be/ejlaQhOCUHw
If anything weird happens like line numbers getting out of sync, try unplugging the printer, kill the server, and reload.
Otherwise, often just disconnecting in the interface works fine.
For Prusa printers, make sure your serial baud rate is 115200 before you select the serial port; for other Marling printers it might be that, or 57600, or the Ultimaker 2 default of 250000.
A short video to get up and running in 2 minutes (assuming you’ve plugged your printer into your computer via USB and turned it on!)
Try these instructions for using the livecoding editor and controlling the printer.
LivePrinter is an open source system for live, immediate drawing and fabrication with 3D printers. It’s particularly useful for:
LivePrinter combines the design and 3D printing of objects into a single, iterative process. Livecoding is used to control manufacturing precisely and procedurally with real-time flexibility. It brings improvisation and intuition to the act of making new computational forms. The system extends digital printing and CNC machining into the realm of performance and also has potential in design and science pedagogy.
The current software toolchains of 3D printing place the artist and designer at a difficult distance from the physical process of making. There is little space for live improvisation and experimentation, especially with key properties that directly affect printing materials, like temperature and print speeds.
You can watch a video of LivePrinter in action, running experimental tests, creating generative shapes, and exploring printing in air:
LivePrinter is designed to be:
Overview of the system:
LivePrinter is part web client and server for livecoding and part 3D printer host application on the back end. On the front end, a user will livecode in a web browser (or other web client). Their code is compiled into machine-appropriate G-Code, packaged into JSON-ROPC calls and sent via AJAX (http) to the back end, where it is sent to the printer via a serial connection. Responses from the printer are sent back to the web client. This whole process is asynchronous, for speed and for tracking which messages successfully were send (and responses received).
More than one person might be involved: the system is designed to support collaborative livecoding and potentially many 3D printers, as complex as that is! (Look at the Max/MSP example in the misc folder for an example of this)
LivePrinter now send useful print events that you can use to build visualisers and collect and download GCode commands. (There’s a public demo on this coming soon)
Add event listeners using the addXXXXListener()
function, sending it an object with a key that is gcodeEvent
, printEvent
, errorEvent
with a function that handles an object like:
const lp = new Printer();
const gcode = [];
const gcodeListener = {
gcodeEvent: (g) => gcode.push(g),
};
lp.addGCodeListener(gcodeListener);
// later on...
lp.removeGCodeListener(gcodeListener);
lp.addPrintListener({
printEvent: ({
type,
newPosition,
oldPosition,
speed,
moveTime,
totalMoveTime,
layerHeight,
length,
}) => {
console.log("Print event:");
console.log("old", oldPosition);
console.log("new", newPosition);
console.log("length", length);
console.log("speed", speed);
},
});
Events from functions:
1. this.printEvent({
type: "drawtime-start",
speed: print speed,
start: start time in ms,
end: estimated end time in ms,
});
2. this.printEvent({
type: "drawtime-end",
speed: print speed,
start: start time in ms,
end: actual end time in ms,
});
printEvent({
type: "draw-start",
speed: current print speed,
length: distance to move,
});
printEvent({
type: "draw-end",
speed: print speed,
length: distance actually moved
});
printEvent({
type: "retract",
speed: retractSpeed,
length: retract Length,
});
this.printEvent({
'type': 'unretract',
'speed':this.retractSpeed,
'length': this.retractLength,
})
extrude event:
printEvent({
type: "extrude",
newPosition: { ...this.position.axes },
oldPosition: { ...oldPosition },
speed: this._printSpeed,
moveTime: moveTime,
totalMoveTime: this.totalMoveTime,
layerHeight: this.layerHeight,
length: distanceMag,
});
travel event:
printEvent({
type: "travel",
newPosition: { ...this.position.axes },
oldPosition: { ...oldPosition },
speed: this._travelSpeed,
moveTime: moveTime,
totalMoveTime: this.totalMoveTime,
layerHeight: this.layerHeight,
length: distanceMag,
});
If you fire up the server and then navigate a browser to http://localhost:8888/jsontest you can play with the API in a hands-on web page.
This is what the API looks like, currently:
The back-end web server and websockets service uses a Tornado server written in Python and a very-adapted version of Cura’s 3D printing driver that connects to a Marlin Firmware based 3D printer (currently tested on an Ultimaker 2+) and on various Prusa printers, as well as homebrew Marlin-based ones.
A more complicated architecture diagram:
If you want to get involved, please give us a shout! The more the better.
There is an API and function reference for the client end which needs formatting work.
It makes sense (in general, not just for this project) to do Python development using isolated virtual environments, to prevent contamination or clashes between modules and utilities by keeping all configuration sandboxed. Check the docs to see how to isntall a virtual environment for your OS, but on OS X and Linux its:
cd LIVEPRINTER_MAIN_FOLDER
cd liveprinter
python3 -m venv ./venv
# create a Python 3 virtual environment
source ./venv/bin/activate
# activate the environment
pip install --upgrade pip
# get the latest pip in the virtual environment (probably not essential)
pip install tornado pyserial tornado_jsonrpc2
# get the libraries we need
python LivePrinterServer.py
# launch the server in the Python 3 sandbox
Then when all is done:
deactivate
# wind down the environment
LivePrinter is completely open source (AGPL 3.0). To summarise, it is built on and inspired by other open source projects:
This project was started by Evan Raskob, artist, livecoder and Course Leader for BSc Computer Science at UAL (previously at Goldsmiths and the Royal College of Art in London), and PhD in arts and Computational Technology from Goldsmiths Department of Computing. If you’d like to get in touch, please email him at evan.raskob@arts.ac.uk.