2023-01-30 16:27:06 +03:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/xml"
|
|
|
|
"fmt"
|
2024-02-08 16:49:01 +03:00
|
|
|
"log/slog"
|
2023-01-30 16:27:06 +03:00
|
|
|
"net/http"
|
2023-09-23 13:05:40 +03:00
|
|
|
"strings"
|
2023-01-30 16:27:06 +03:00
|
|
|
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
2023-09-23 13:05:40 +03:00
|
|
|
"rsc.io/qr"
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2024-12-03 00:03:18 +03:00
|
|
|
"github.com/mjl-/mox/admin"
|
2023-01-30 16:27:06 +03:00
|
|
|
"github.com/mjl-/mox/smtp"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
metricAutoconf = promauto.NewCounterVec(
|
|
|
|
prometheus.CounterOpts{
|
|
|
|
Name: "mox_autoconf_request_total",
|
|
|
|
Help: "Number of autoconf requests.",
|
|
|
|
},
|
|
|
|
[]string{"domain"},
|
|
|
|
)
|
|
|
|
metricAutodiscover = promauto.NewCounterVec(
|
|
|
|
prometheus.CounterOpts{
|
|
|
|
Name: "mox_autodiscover_request_total",
|
|
|
|
Help: "Number of autodiscover requests.",
|
|
|
|
},
|
|
|
|
[]string{"domain"},
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
// Autoconfiguration/Autodiscovery:
|
|
|
|
//
|
|
|
|
// - Thunderbird will request an "autoconfig" xml file.
|
|
|
|
// - Microsoft tools will request an "autodiscovery" xml file.
|
|
|
|
// - In my tests on an internal domain, iOS mail only talks to Apple servers, then
|
2023-09-23 13:05:40 +03:00
|
|
|
// does not attempt autoconfiguration. Possibly due to them being private DNS
|
|
|
|
// names. Apple software can be provisioned with "mobileconfig" profile files,
|
|
|
|
// which users can download after logging in.
|
2023-01-30 16:27:06 +03:00
|
|
|
//
|
|
|
|
// DNS records seem optional, but autoconfig.<domain> and autodiscover.<domain>
|
|
|
|
// (both CNAME or A) are useful, and so is SRV _autodiscovery._tcp.<domain> 0 0 443
|
|
|
|
// autodiscover.<domain> (or just <hostname> directly).
|
|
|
|
//
|
|
|
|
// Autoconf/discovery only works with valid TLS certificates, not with self-signed
|
|
|
|
// certs. So use it on public endpoints with certs signed by common CA's, or run
|
|
|
|
// your own (internal) CA and import the CA cert on your devices.
|
|
|
|
//
|
|
|
|
// Also see https://roll.urown.net/server/mail/autoconfig.html
|
|
|
|
|
|
|
|
// Autoconfiguration for Mozilla Thunderbird.
|
|
|
|
// User should create a DNS record: autoconfig.<domain> (CNAME or A).
|
|
|
|
// See https://wiki.mozilla.org/Thunderbird:Autoconfiguration:ConfigFileFormat
|
2023-03-04 02:49:02 +03:00
|
|
|
func autoconfHandle(w http.ResponseWriter, r *http.Request) {
|
2023-12-05 15:35:58 +03:00
|
|
|
log := pkglog.WithContext(r.Context())
|
2023-03-04 02:49:02 +03:00
|
|
|
|
|
|
|
var addrDom string
|
|
|
|
defer func() {
|
|
|
|
metricAutoconf.WithLabelValues(addrDom).Inc()
|
|
|
|
}()
|
|
|
|
|
|
|
|
email := r.FormValue("emailaddress")
|
2023-12-05 15:35:58 +03:00
|
|
|
log.Debug("autoconfig request", slog.String("email", email))
|
2023-03-04 02:49:02 +03:00
|
|
|
addr, err := smtp.ParseAddress(email)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "400 - bad request - invalid parameter emailaddress", http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2024-12-03 00:03:18 +03:00
|
|
|
socketType := func(tlsMode admin.TLSMode) (string, error) {
|
2023-09-23 13:05:40 +03:00
|
|
|
switch tlsMode {
|
2024-12-03 00:03:18 +03:00
|
|
|
case admin.TLSModeImmediate:
|
2023-09-23 13:05:40 +03:00
|
|
|
return "SSL", nil
|
2024-12-03 00:03:18 +03:00
|
|
|
case admin.TLSModeSTARTTLS:
|
2023-09-23 13:05:40 +03:00
|
|
|
return "STARTTLS", nil
|
2024-12-03 00:03:18 +03:00
|
|
|
case admin.TLSModeNone:
|
2023-09-23 13:05:40 +03:00
|
|
|
return "plain", nil
|
|
|
|
default:
|
|
|
|
return "", fmt.Errorf("unknown tls mode %v", tlsMode)
|
|
|
|
}
|
2023-03-04 02:49:02 +03:00
|
|
|
}
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-09-23 13:05:40 +03:00
|
|
|
var imapTLS, submissionTLS string
|
2024-12-03 00:03:18 +03:00
|
|
|
config, err := admin.ClientConfigDomain(addr.Domain)
|
2023-09-23 13:05:40 +03:00
|
|
|
if err == nil {
|
|
|
|
imapTLS, err = socketType(config.IMAP.TLSMode)
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
submissionTLS, err = socketType(config.Submission.TLSMode)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "400 - bad request - "+err.Error(), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-03-04 02:49:02 +03:00
|
|
|
// Thunderbird doesn't seem to allow U-labels, always return ASCII names.
|
|
|
|
var resp autoconfigResponse
|
|
|
|
resp.Version = "1.1"
|
|
|
|
resp.EmailProvider.ID = addr.Domain.ASCII
|
|
|
|
resp.EmailProvider.Domain = addr.Domain.ASCII
|
|
|
|
resp.EmailProvider.DisplayName = email
|
|
|
|
resp.EmailProvider.DisplayShortName = addr.Domain.ASCII
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-03-04 02:49:02 +03:00
|
|
|
// todo: specify SCRAM-SHA-256 once thunderbird and autoconfig supports it. or perhaps that will fall under "password-encrypted" by then.
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-03-04 02:49:02 +03:00
|
|
|
resp.EmailProvider.IncomingServer.Type = "imap"
|
2023-09-23 13:05:40 +03:00
|
|
|
resp.EmailProvider.IncomingServer.Hostname = config.IMAP.Host.ASCII
|
|
|
|
resp.EmailProvider.IncomingServer.Port = config.IMAP.Port
|
|
|
|
resp.EmailProvider.IncomingServer.SocketType = imapTLS
|
2023-03-04 02:49:02 +03:00
|
|
|
resp.EmailProvider.IncomingServer.Username = email
|
|
|
|
resp.EmailProvider.IncomingServer.Authentication = "password-encrypted"
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-03-04 02:49:02 +03:00
|
|
|
resp.EmailProvider.OutgoingServer.Type = "smtp"
|
2023-09-23 13:05:40 +03:00
|
|
|
resp.EmailProvider.OutgoingServer.Hostname = config.Submission.Host.ASCII
|
|
|
|
resp.EmailProvider.OutgoingServer.Port = config.Submission.Port
|
|
|
|
resp.EmailProvider.OutgoingServer.SocketType = submissionTLS
|
2023-03-04 02:49:02 +03:00
|
|
|
resp.EmailProvider.OutgoingServer.Username = email
|
|
|
|
resp.EmailProvider.OutgoingServer.Authentication = "password-encrypted"
|
|
|
|
|
|
|
|
// todo: should we put the email address in the URL?
|
2023-09-23 13:05:40 +03:00
|
|
|
resp.ClientConfigUpdate.URL = fmt.Sprintf("https://autoconfig.%s/mail/config-v1.1.xml", addr.Domain.ASCII)
|
2023-03-04 02:49:02 +03:00
|
|
|
|
|
|
|
w.Header().Set("Content-Type", "application/xml; charset=utf-8")
|
|
|
|
enc := xml.NewEncoder(w)
|
|
|
|
enc.Indent("", "\t")
|
|
|
|
fmt.Fprint(w, xml.Header)
|
|
|
|
if err := enc.Encode(resp); err != nil {
|
|
|
|
log.Errorx("marshal autoconfig response", err)
|
2023-01-30 16:27:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Autodiscover from Microsoft, also used by Thunderbird.
|
2023-10-13 09:16:46 +03:00
|
|
|
// User should create a DNS record: _autodiscover._tcp.<domain> SRV 0 0 443 <hostname>
|
2023-02-05 18:29:03 +03:00
|
|
|
//
|
|
|
|
// In practice, autodiscover does not seem to work wit microsoft clients. A
|
|
|
|
// connectivity test tool for outlook is available on
|
|
|
|
// https://testconnectivity.microsoft.com/, it has an option to do "Autodiscover to
|
|
|
|
// detect server settings". Incoming TLS connections are all failing, with various
|
|
|
|
// errors.
|
|
|
|
//
|
|
|
|
// Thunderbird does understand autodiscover.
|
2023-03-04 02:49:02 +03:00
|
|
|
func autodiscoverHandle(w http.ResponseWriter, r *http.Request) {
|
2023-12-05 15:35:58 +03:00
|
|
|
log := pkglog.WithContext(r.Context())
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-03-04 02:49:02 +03:00
|
|
|
var addrDom string
|
|
|
|
defer func() {
|
|
|
|
metricAutodiscover.WithLabelValues(addrDom).Inc()
|
|
|
|
}()
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-03-04 02:49:02 +03:00
|
|
|
if r.Method != "POST" {
|
|
|
|
http.Error(w, "405 - method not allowed - post required", http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-03-04 02:49:02 +03:00
|
|
|
var req autodiscoverRequest
|
|
|
|
if err := xml.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
http.Error(w, "400 - bad request - parsing autodiscover request: "+err.Error(), http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-12-05 15:35:58 +03:00
|
|
|
log.Debug("autodiscover request", slog.String("email", req.Request.EmailAddress))
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-03-04 02:49:02 +03:00
|
|
|
addr, err := smtp.ParseAddress(req.Request.EmailAddress)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "400 - bad request - invalid parameter emailaddress", http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
2023-01-30 16:27:06 +03:00
|
|
|
|
2023-09-23 13:05:40 +03:00
|
|
|
// tlsmode returns the "ssl" and "encryption" fields.
|
2024-12-03 00:03:18 +03:00
|
|
|
tlsmode := func(tlsMode admin.TLSMode) (string, string, error) {
|
2023-09-23 13:05:40 +03:00
|
|
|
switch tlsMode {
|
2024-12-03 00:03:18 +03:00
|
|
|
case admin.TLSModeImmediate:
|
2023-09-23 13:05:40 +03:00
|
|
|
return "on", "TLS", nil
|
2024-12-03 00:03:18 +03:00
|
|
|
case admin.TLSModeSTARTTLS:
|
2023-09-23 13:05:40 +03:00
|
|
|
return "on", "", nil
|
2024-12-03 00:03:18 +03:00
|
|
|
case admin.TLSModeNone:
|
2023-09-23 13:05:40 +03:00
|
|
|
return "off", "", nil
|
|
|
|
default:
|
|
|
|
return "", "", fmt.Errorf("unknown tls mode %v", tlsMode)
|
|
|
|
}
|
2023-03-04 02:49:02 +03:00
|
|
|
}
|
|
|
|
|
2023-09-23 13:05:40 +03:00
|
|
|
var imapSSL, imapEncryption string
|
|
|
|
var submissionSSL, submissionEncryption string
|
2024-12-03 00:03:18 +03:00
|
|
|
config, err := admin.ClientConfigDomain(addr.Domain)
|
2023-09-23 13:05:40 +03:00
|
|
|
if err == nil {
|
|
|
|
imapSSL, imapEncryption, err = tlsmode(config.IMAP.TLSMode)
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
submissionSSL, submissionEncryption, err = tlsmode(config.Submission.TLSMode)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "400 - bad request - "+err.Error(), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
2023-03-04 02:49:02 +03:00
|
|
|
|
|
|
|
// The docs are generated and fragmented in many tiny pages, hard to follow.
|
|
|
|
// High-level starting point, https://learn.microsoft.com/en-us/openspecs/exchange_server_protocols/ms-oxdscli/78530279-d042-4eb0-a1f4-03b18143cd19
|
|
|
|
// Request: https://learn.microsoft.com/en-us/openspecs/exchange_server_protocols/ms-oxdscli/2096fab2-9c3c-40b9-b123-edf6e8d55a9b
|
|
|
|
// Response, protocol: https://learn.microsoft.com/en-us/openspecs/exchange_server_protocols/ms-oxdscli/f4238db6-a983-435c-807a-b4b4a624c65b
|
|
|
|
// It appears autodiscover does not allow specifying SCRAM-SHA-256 as
|
|
|
|
// authentication method, or any authentication method that real clients actually
|
|
|
|
// use. See
|
|
|
|
// https://learn.microsoft.com/en-us/openspecs/exchange_server_protocols/ms-oxdscli/21fd2dd5-c4ee-485b-94fb-e7db5da93726
|
|
|
|
|
|
|
|
w.Header().Set("Content-Type", "application/xml; charset=utf-8")
|
|
|
|
|
|
|
|
resp := autodiscoverResponse{}
|
|
|
|
resp.XMLName.Local = "Autodiscover"
|
|
|
|
resp.XMLName.Space = "http://schemas.microsoft.com/exchange/autodiscover/responseschema/2006"
|
|
|
|
resp.Response.XMLName.Local = "Response"
|
|
|
|
resp.Response.XMLName.Space = "http://schemas.microsoft.com/exchange/autodiscover/outlook/responseschema/2006a"
|
|
|
|
resp.Response.Account = autodiscoverAccount{
|
|
|
|
AccountType: "email",
|
|
|
|
Action: "settings",
|
|
|
|
Protocol: []autodiscoverProtocol{
|
|
|
|
{
|
|
|
|
Type: "IMAP",
|
2023-09-23 13:05:40 +03:00
|
|
|
Server: config.IMAP.Host.ASCII,
|
|
|
|
Port: config.IMAP.Port,
|
2023-03-04 02:49:02 +03:00
|
|
|
LoginName: req.Request.EmailAddress,
|
|
|
|
SSL: imapSSL,
|
|
|
|
Encryption: imapEncryption,
|
|
|
|
SPA: "off", // Override default "on", this is Microsofts proprietary authentication protocol.
|
|
|
|
AuthRequired: "on",
|
2023-01-30 16:27:06 +03:00
|
|
|
},
|
2023-03-04 02:49:02 +03:00
|
|
|
{
|
|
|
|
Type: "SMTP",
|
2023-09-23 13:05:40 +03:00
|
|
|
Server: config.Submission.Host.ASCII,
|
|
|
|
Port: config.Submission.Port,
|
2023-03-04 02:49:02 +03:00
|
|
|
LoginName: req.Request.EmailAddress,
|
2023-09-23 13:05:40 +03:00
|
|
|
SSL: submissionSSL,
|
|
|
|
Encryption: submissionEncryption,
|
2023-03-04 02:49:02 +03:00
|
|
|
SPA: "off", // Override default "on", this is Microsofts proprietary authentication protocol.
|
|
|
|
AuthRequired: "on",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
enc := xml.NewEncoder(w)
|
|
|
|
enc.Indent("", "\t")
|
|
|
|
fmt.Fprint(w, xml.Header)
|
|
|
|
if err := enc.Encode(resp); err != nil {
|
|
|
|
log.Errorx("marshal autodiscover response", err)
|
2023-01-30 16:27:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Thunderbird requests these URLs for autoconfig/autodiscover:
|
|
|
|
// https://autoconfig.example.org/mail/config-v1.1.xml?emailaddress=user%40example.org
|
|
|
|
// https://autodiscover.example.org/autodiscover/autodiscover.xml
|
|
|
|
// https://example.org/.well-known/autoconfig/mail/config-v1.1.xml?emailaddress=user%40example.org
|
|
|
|
// https://example.org/autodiscover/autodiscover.xml
|
|
|
|
type autoconfigResponse struct {
|
|
|
|
XMLName xml.Name `xml:"clientConfig"`
|
|
|
|
Version string `xml:"version,attr"`
|
|
|
|
|
|
|
|
EmailProvider struct {
|
|
|
|
ID string `xml:"id,attr"`
|
|
|
|
Domain string `xml:"domain"`
|
|
|
|
DisplayName string `xml:"displayName"`
|
|
|
|
DisplayShortName string `xml:"displayShortName"`
|
|
|
|
|
|
|
|
IncomingServer struct {
|
|
|
|
Type string `xml:"type,attr"`
|
|
|
|
Hostname string `xml:"hostname"`
|
|
|
|
Port int `xml:"port"`
|
|
|
|
SocketType string `xml:"socketType"`
|
|
|
|
Username string `xml:"username"`
|
|
|
|
Authentication string `xml:"authentication"`
|
|
|
|
} `xml:"incomingServer"`
|
|
|
|
|
|
|
|
OutgoingServer struct {
|
|
|
|
Type string `xml:"type,attr"`
|
|
|
|
Hostname string `xml:"hostname"`
|
|
|
|
Port int `xml:"port"`
|
|
|
|
SocketType string `xml:"socketType"`
|
|
|
|
Username string `xml:"username"`
|
|
|
|
Authentication string `xml:"authentication"`
|
|
|
|
} `xml:"outgoingServer"`
|
|
|
|
} `xml:"emailProvider"`
|
|
|
|
|
|
|
|
ClientConfigUpdate struct {
|
|
|
|
URL string `xml:"url,attr"`
|
|
|
|
} `xml:"clientConfigUpdate"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type autodiscoverRequest struct {
|
|
|
|
XMLName xml.Name `xml:"Autodiscover"`
|
|
|
|
Request struct {
|
|
|
|
EmailAddress string `xml:"EMailAddress"`
|
|
|
|
AcceptableResponseSchema string `xml:"AcceptableResponseSchema"`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type autodiscoverResponse struct {
|
|
|
|
XMLName xml.Name
|
|
|
|
Response struct {
|
|
|
|
XMLName xml.Name
|
|
|
|
Account autodiscoverAccount
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type autodiscoverAccount struct {
|
|
|
|
AccountType string
|
|
|
|
Action string
|
|
|
|
Protocol []autodiscoverProtocol
|
|
|
|
}
|
|
|
|
|
|
|
|
type autodiscoverProtocol struct {
|
|
|
|
Type string
|
|
|
|
Server string
|
|
|
|
Port int
|
|
|
|
DirectoryPort int
|
|
|
|
ReferralPort int
|
|
|
|
LoginName string
|
|
|
|
SSL string
|
|
|
|
Encryption string `xml:",omitempty"`
|
|
|
|
SPA string
|
|
|
|
AuthRequired string
|
|
|
|
}
|
2023-09-23 13:05:40 +03:00
|
|
|
|
|
|
|
// Serve a .mobileconfig file. This endpoint is not a standard place where Apple
|
|
|
|
// devices look. We point to it from the account page.
|
|
|
|
func mobileconfigHandle(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if r.Method != "GET" {
|
|
|
|
http.Error(w, "405 - method not allowed - get required", http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
2023-09-23 18:50:32 +03:00
|
|
|
addresses := r.FormValue("addresses")
|
2023-09-23 13:05:40 +03:00
|
|
|
fullName := r.FormValue("name")
|
2023-09-23 18:50:32 +03:00
|
|
|
var buf []byte
|
|
|
|
var err error
|
|
|
|
if addresses == "" {
|
|
|
|
err = fmt.Errorf("missing/empty field addresses")
|
|
|
|
}
|
|
|
|
l := strings.Split(addresses, ",")
|
|
|
|
if err == nil {
|
|
|
|
buf, err = MobileConfig(l, fullName)
|
|
|
|
}
|
2023-09-23 13:05:40 +03:00
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "400 - bad request - "+err.Error(), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
h := w.Header()
|
2023-09-23 18:50:32 +03:00
|
|
|
filename := l[0]
|
2023-09-23 13:05:40 +03:00
|
|
|
filename = strings.ReplaceAll(filename, ".", "-")
|
|
|
|
filename = strings.ReplaceAll(filename, "@", "-at-")
|
|
|
|
filename = "email-account-" + filename + ".mobileconfig"
|
|
|
|
h.Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"`, filename))
|
|
|
|
w.Write(buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Serve a png file with qrcode with the link to the .mobileconfig file, should be
|
|
|
|
// helpful for mobile devices.
|
|
|
|
func mobileconfigQRCodeHandle(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if r.Method != "GET" {
|
|
|
|
http.Error(w, "405 - method not allowed - get required", http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if !strings.HasSuffix(r.URL.Path, ".qrcode.png") {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Compose URL, scheme and host are not set.
|
|
|
|
u := *r.URL
|
|
|
|
if r.TLS == nil {
|
|
|
|
u.Scheme = "http"
|
|
|
|
} else {
|
|
|
|
u.Scheme = "https"
|
|
|
|
}
|
|
|
|
u.Host = r.Host
|
|
|
|
u.Path = strings.TrimSuffix(u.Path, ".qrcode.png")
|
|
|
|
|
|
|
|
code, err := qr.Encode(u.String(), qr.L)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, "500 - internal server error - generating qr-code: "+err.Error(), http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
h := w.Header()
|
|
|
|
h.Set("Content-Type", "image/png")
|
|
|
|
w.Write(code.PNG())
|
|
|
|
}
|