Use DYMO LetraTag LT-200B thermal label printer over Bluetooth in Python, without depending on its app.
I was not expecting to own a label printer, but after caming across to this online, I thought why I shouldn't get it for whatever reason since it was affordable. After a while, I decided to tinker with this printer so I can print anything whatever I would like instead of being limited with its mobile app, so here it is.
The image on the printed label shown in the picture can be found here.
More about this printer model
This printer doesn't use inks, it instead writes on the labels with thermal. And obviously, it does require a label cartridge (or called "casette"). From what I've seen, all LetraTag branded label casettes have the same same shape and dimensions, so it is easy to get another if the casette that came with the printer runs out. There are different kind of labels (paper, plastic, metalic etc.) available, and all of them are self-adhesive. The casettes don't have an electronical part, which makes the printer to be able to use casettes manufactured by someone else.
The label dimensions are 12 millimeter in height, however the printable area is just 32 pixels, so labels will have an very obvious white space even if all printable space is filled with black. The print resolution is 200 DPI and, as stated on the original website, it has a print speed of up to 7 millimeters per second. So unfortunately the print resolution is not perfect for everything, but still, it does its job, and in the end, I enjoyed playing around with this printer, even if it was too simple.
python -m pip install --upgrade https://github.com/ysfchn/dymo-bluetooth/archive/refs/heads/main.zip
Python 3.10 or up is targeted, but 3.9 should work too. It depends on bleak
for Bluetooth communication, and pillow
for importing images from files. If python-barcode
is installed, it can be used to print barcodes.
There is a small CLI to print image files in a nearby printer.
python -m dymo_bluetooth --help
Or in a Python program:
from dymo_bluetooth import discover_printers, Canvas
import asyncio
async def main():
# Canvas class allows constructing 1-bit images in ease.
canvas = Canvas()
# Pixels can be set and retrieved with methods.
canvas.set_pixel(0, 0, True)
# Images needs to be stretched by at least 2 times like
# how its mobile app does, otherwise printer will render
# the labels too thin.
canvas = canvas.stretch_image(2)
# Get a list of printers.
printers = await discover_printers()
# Get the first found printer and print the
# constructed Canvas. Returns the print status.
await printers[0].print(canvas)
asyncio.run(main())
This program is licensed under MIT License.
It is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See MIT license for details.
This program is not supported, sponsored, affiliated, approved, or endorsed in any way with DYMO. LetraTag is an trademark of DYMO. All other mentioned trademarks are the property of respective owners.
This printer uses Bluetooth LE (GATT) to send labels & retrieve its status. It does uses this service UUIDs:
UUID | Used for |
---|---|
be3dd650-2b3d-42f1-99c1-f0f749dd0678 |
Service |
be3dd651-2b3d-42f1-99c1-f0f749dd0678 |
Print request (write -> printer) |
be3dd652-2b3d-42f1-99c1-f0f749dd0678 |
Print reply (notify <- printer) |
be3dd653-2b3d-42f1-99c1-f0f749dd0678 |
Unknown |
2b3d
part in the UUID may not be future-proof, but so far all produced printers of the same model seems to share the same UUID.
To discover the nearby printers, filter for the service UUID or/and check for the MAC address range. This printer has a MAC address in range of 58:CF:79:00:00:00
and 58:CF:79:FF:FF:FF
, which this MAC range block seems to be owned by Espressif Inc. according to Bluetooth vendor list.
All data structures are explained below so it can be used as a future reference.
Directives are simply command types that found in the payload. A directive directly follows another directive, and so on. A payload may contain more than one directive.
Each directive consists of 2 bytes, first byte is always an ASCII escape character 1B
(in hexadecimal, or 27 in decimal), and second byte is the ASCII code point of the character that specifies the command type.
COMMAND_TYPE[1] = 73 | 4D | 43 | 44 | 45 | 41 | 51
DIRECTIVE[2] = 1B + COMMAND_TYPE[1]
Command | Character | = Code point (hex) | = Code point (decimal) | Notes |
---|---|---|---|---|
START |
s |
0x73 |
115 |
Start command. Each payload starts with this directive. |
MEDIA_TYPE |
M |
0x4d |
77 |
? |
PRINT_DENSITY |
C |
0x43 |
67 |
Seems unused. |
PRINT_DATA |
D |
0x44 |
68 |
Defines the image that will be printed, see below for byte format. |
FORM_FEED |
E |
0x45 |
69 |
Follows after PRINT_DATA . |
STATUS |
A |
0x41 |
65 |
Requests the status (?) |
END |
Q |
0x51 |
81 |
End command. Each payload ends with this directive. |
Below is the structure of each command that follows after its directive:
Each payload starts with directive. This command follows with 4 constant bytes after its directive.
START[6] = DIRECTIVE[2] + 9A 02 00 00
This command follows with 1 byte containing a number of some type (?) to be set for the printer. It doesn't really seem to be used by this printer, but this command was found in its mobile app anyway.
When this number has set to anything, the printer should advertise this number when searching for Bluetooth devices nearby. (not tested) So I guess it was planned to be used for some sort of casette checking & changing casette type, so printer can hold this value even when its turned off.
MEDIA_TYPE[3] = DIRECTIVE[2] + ??
Seems unused by the printer. Not sure what it does.
PRINT_DENSITY[?] = DIRECTIVE[2] + ...
Used after PRINT_DATA
command. Doesn't take additional bytes other than its directive.
FORM_FEED[2] = DIRECTIVE[2]
Used after FORM_FEED
command. Doesn't take additional bytes other than its directive.
STATUS[2] = DIRECTIVE[2]
Each payload ends with directive. Doesn't take additional bytes other than its directive.
END[2] = DIRECTIVE[2]
This command defines the data of the image that will be printed to the label. It is followed by:
-
Bits per pixel value; this is always equals to
1
, so in hex that would be0x01
. -
Alignment; this is always equals to
2
, so in hex that would be0x02
. This is always seems the case regardless of the content of the image, so not sure how printer will react for values other than2
. -
Width of the image. This value consists of 4 bytes, in little-endian. Since the label is printed horizontally, it might be changed based on your input image, so continue reading about image format.
-
Height of the image. This value consists of 4 bytes, in little-endian. Since labels can only contain 32 pixels from top to the bottom, simply use 32 as a height, so in hexadecimal, that would be:
20 00 00 00
-
Image data, see "Image format".
# Printer have fixed 32 pixel size
HEIGHT[4] = 20 00 00 00
PRINT_DATA[variable] = DIRECTIVE[2] + 01 02 + WIDTH[4] + HEIGHT[4] + IMAGE_DATA[4 * WIDTH]
The printer accepts images in portrait direction, formatted as 1-bit monochrome and stored in little endian. To put it simply, it uses bits (1
and 0
) to set each pixel, and since a byte contains 8 bits, the printer will require 4 bytes (4 * 8 = 32 pixels, which is the height of the printable area) to print a full line from top to the bottom.
8 bits 4 bytes = 32 bits (8 * 4) = 1 printing line
┌─────────────┐ ┌─────────┐
│ │ │ │
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 -> 00 00 00 00 -> 1 full empty line
binary binary binary binary hex
1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 -> FF FF FF FF -> 1 full black line
binary binary binary binary hex
Since the format is in little endian, the first left-top pixel (X = 0, Y = 0) is inserted in the last byte of 4 bytes, and the biggest bit in that byte set.
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 -> 00 00 00 80 -> 1 pixel in [x = 0, y = 0]
binary binary binary │ binary hex
│
0 0 0 └─── 128
decimal decimal decimal decimal
As explained above, line always take up 4 bytes, so the image width can be found by dividing the byte count to 4. And since the image height doesn't vary (labels are printed horizontally in this printer, from left to right) the image height is always 32 pixels.
Each line is directly followed by another line (if any), so there is no additional separator between lines, and it is ordered sequantally, so the first 4 bytes in the image data will be the first line that will be printed out on the label.
There are 2 payloads that can be sent to the printer. Each payload starts with an HEADER
, and it does follow with the payload data.
Payload | Requires chunking? | Usage |
---|---|---|
Print request | Yes | Sends an image to be printed out on the label. |
Set media type | No | ? |
Magic value consists of 2 bytes, which is 12 34
(in hexadecimal) or [18, 52] in decimal.
MAGIC[2] = 12 34
Header consists of 9 bytes in total. First two bytes are constant, which is FF F0
. Third and forth byte is MAGIC
, which is explained above. And it is followed by the length of the payload stored in 4 bytes (little-endian) that is going to follow after this header.
The ninth byte of the header is the checksum of the header, and it is calculated by simply getting sum of the all previous 8 bytes and getting the first byte of the sum.
PARTIAL_HEADER[8] = FF F0 + MAGIC[2] + PAYLOAD_LENGTH[4]
CHECKSUM = SUM(PARTIAL_HEADER) & 0xFF
HEADER[9] = PARTIAL_HEADER[8] + CHECKSUM
A print request payload consists of these directives; START
, PRINT_DATA
, FORM_FEED
, STATUS
and END
respectively. Directives follow after HEADER
, so it would be the first.
PRINT_REQUEST[variable] = HEADER[9] + START[6] + PRINT_DATA[variable] + FORM_FEED[2] + STATUS[2] + END[2]
Due to both the way the printer works and the way the Bluetooth itself work, this payload needs to be broken down into chunks in each 500 bytes. Each chunk must be send separately over Bluetooth, the printer will start printing until it receives the last chunk.
Chunking needs to be start right after HEADER
, so only the header is sent first (which is the 9 bytes), then the each chunk is separately sent via Bluetooth. So number of the total Bluetooth writes would be 1 + chunk count
.
Each chunk must contain its index before chunk data itself followed by. The chunk data can be smaller than 500, but it can only contain maximum 500 bytes, inclusive. In the end, you will have the full chunk to be sent over Bluetooth which contains the index and the data.
CHUNK[0...501] = CHUNK_INDEX[1] + CHUNK_DATA[0...500]
The last chunk must contain the MAGIC
value appended end of the chunk.
CHUNK[0...503] = CHUNK_INDEX[1] + CHUNK_DATA[0...500] + MAGIC[2]
A media type set payload consists of these directives; START
, MEDIA_TYPE
and END
respectively. Directives follow after HEADER
, so it would be the first.
SET_MEDIA_TYPE[20] = HEADER[9] + START[6] + MEDIA_TYPE[3] + END[2]
This payload doesn't require chunking and can be sent with single Bluetooth transaction.
Printer will notify on the print reply UUID, giving the status of the printing operation. First byte is always an ASCII escape character 1B
(in hexadecimal, or 27 in decimal), and second byte is the ASCII code point of the R
character (0x52
in hexadecimal, or 82 in decimal) that stands for the result.
Third byte represents the status code. The printer may be inconsistent when it comes to the status codes, see Result
enum in the code for more details about available status codes and their (supposedly) meanings.
RESULT[3] = 1B 52 + STATUS[1]
The printer will automatically print its MAC address if power button was pressed long.