1

Can I print directly to my (physical, external) printer from Golang, without using printer drivers or CUPS or any other such complexity?

Alex Flint
  • 6,040
  • 8
  • 41
  • 80

1 Answers1

7

Yes! The following prints a postscript file from Golang using IPP (the internet printing protocol), which most network printers manufactured in the past 20 years support:

import (
    "bytes"
    "fmt"
    "io"
    "io/ioutil"
    "net/http"
    "os"
    "strconv"

    "github.com/alexflint/go-arg"
    "github.com/kr/pretty"
    "github.com/phin1x/go-ipp"
)

func Main() error {
    var args struct {
        URI            string `arg:"positional,required"`
        PostscriptFile string `arg:"positional,required"`
    }
    arg.MustParse(&args)

    // define a ipp request
    req := ipp.NewRequest(ipp.OperationPrintJob, 1)
    req.OperationAttributes[ipp.AttributeCharset] = "utf-8"
    req.OperationAttributes[ipp.AttributeNaturalLanguage] = "en"
    req.OperationAttributes[ipp.AttributePrinterURI] = args.URI
    req.OperationAttributes[ipp.AttributeRequestingUserName] = "some-user"
    req.OperationAttributes[ipp.AttributeDocumentFormat] = "application/octet-stream"

    // encode request to bytes
    payload, err := req.Encode()
    if err != nil {
        return fmt.Errorf("error encoding ipp request: %w", err)
    }

    // read the test page
    postscript, err := ioutil.ReadFile(args.PostscriptFile)
    if err != nil {
        return fmt.Errorf("error reading postscript file: %w", err)
    }
    payload = append(payload, postscript...)

    // send ipp request to remote server via http
    httpReq, err := http.NewRequest("POST", args.URI, bytes.NewReader(payload))
    if err != nil {
        return fmt.Errorf("error creating http request: %w", err)
    }

    // set ipp headers
    httpReq.Header.Set("Content-Length", strconv.Itoa(len(payload)))
    httpReq.Header.Set("Content-Type", ipp.ContentTypeIPP)

    // perform the request
    var httpClient http.Client
    httpResp, err := httpClient.Do(httpReq)
    if err != nil {
        return fmt.Errorf("error executing http request: %w", err)
    }
    defer httpResp.Body.Close()

    // read the response
    buf, err := io.ReadAll(httpResp.Body)
    if err != nil {
        return fmt.Errorf("error reading response body: %w", err)
    }

    // response must be 200 for a successful operation
    // other possible http codes are:
    // - 500 -> server error
    // - 426 -> sever requests a encrypted connection
    // - 401 -> forbidden -> need authorization header or user is not permitted
    if httpResp.StatusCode != 200 {
        return fmt.Errorf("printer said %d: %s", httpResp.StatusCode, buf)
    }

    // decode ipp response
    resp, err := ipp.NewResponseDecoder(bytes.NewReader(buf)).Decode(nil)
    if err != nil {
        return fmt.Errorf("error decoding ipp response: %w", err)
    }

    // print the response
    fmt.Println("Submitted print job. Response was:")
    pretty.Println(resp)
    return nil
}

The URL to use is just http://ip-address-of-myprinter (this was tested on a brother HL-series printer)

Alex Flint
  • 6,040
  • 8
  • 41
  • 80