mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-11-03 19:17:29 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			519 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			519 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Forked Jan. 2015 from http://bitbucket.org/PinIdea/fcgi_client
 | 
						|
// (which is forked from https://code.google.com/p/go-fastcgi-client/)
 | 
						|
 | 
						|
// This fork contains several fixes and improvements by Matt Holt and
 | 
						|
// other contributors to this project.
 | 
						|
 | 
						|
// Copyright 2012 Junqing Tan <ivan@mysqlab.net> and The Go Authors
 | 
						|
// Use of this source code is governed by a BSD-style
 | 
						|
// Part of source code is from Go fcgi package
 | 
						|
 | 
						|
package fastcgi
 | 
						|
 | 
						|
import (
 | 
						|
	"bufio"
 | 
						|
	"bytes"
 | 
						|
	"encoding/binary"
 | 
						|
	"errors"
 | 
						|
	"io"
 | 
						|
	"io/ioutil"
 | 
						|
	"mime/multipart"
 | 
						|
	"net"
 | 
						|
	"net/http"
 | 
						|
	"net/http/httputil"
 | 
						|
	"net/textproto"
 | 
						|
	"net/url"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"strconv"
 | 
						|
	"strings"
 | 
						|
	"sync"
 | 
						|
)
 | 
						|
 | 
						|
const FCGI_LISTENSOCK_FILENO uint8 = 0
 | 
						|
const FCGI_HEADER_LEN uint8 = 8
 | 
						|
const VERSION_1 uint8 = 1
 | 
						|
const FCGI_NULL_REQUEST_ID uint8 = 0
 | 
						|
const FCGI_KEEP_CONN uint8 = 1
 | 
						|
const doubleCRLF = "\r\n\r\n"
 | 
						|
 | 
						|
const (
 | 
						|
	FCGI_BEGIN_REQUEST uint8 = iota + 1
 | 
						|
	FCGI_ABORT_REQUEST
 | 
						|
	FCGI_END_REQUEST
 | 
						|
	FCGI_PARAMS
 | 
						|
	FCGI_STDIN
 | 
						|
	FCGI_STDOUT
 | 
						|
	FCGI_STDERR
 | 
						|
	FCGI_DATA
 | 
						|
	FCGI_GET_VALUES
 | 
						|
	FCGI_GET_VALUES_RESULT
 | 
						|
	FCGI_UNKNOWN_TYPE
 | 
						|
	FCGI_MAXTYPE = FCGI_UNKNOWN_TYPE
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	FCGI_RESPONDER uint8 = iota + 1
 | 
						|
	FCGI_AUTHORIZER
 | 
						|
	FCGI_FILTER
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	FCGI_REQUEST_COMPLETE uint8 = iota
 | 
						|
	FCGI_CANT_MPX_CONN
 | 
						|
	FCGI_OVERLOADED
 | 
						|
	FCGI_UNKNOWN_ROLE
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	FCGI_MAX_CONNS  string = "MAX_CONNS"
 | 
						|
	FCGI_MAX_REQS   string = "MAX_REQS"
 | 
						|
	FCGI_MPXS_CONNS string = "MPXS_CONNS"
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	maxWrite = 65500 // 65530 may work, but for compatibility
 | 
						|
	maxPad   = 255
 | 
						|
)
 | 
						|
 | 
						|
type header struct {
 | 
						|
	Version       uint8
 | 
						|
	Type          uint8
 | 
						|
	Id            uint16
 | 
						|
	ContentLength uint16
 | 
						|
	PaddingLength uint8
 | 
						|
	Reserved      uint8
 | 
						|
}
 | 
						|
 | 
						|
// for padding so we don't have to allocate all the time
 | 
						|
// not synchronized because we don't care what the contents are
 | 
						|
var pad [maxPad]byte
 | 
						|
 | 
						|
func (h *header) init(recType uint8, reqID uint16, contentLength int) {
 | 
						|
	h.Version = 1
 | 
						|
	h.Type = recType
 | 
						|
	h.Id = reqID
 | 
						|
	h.ContentLength = uint16(contentLength)
 | 
						|
	h.PaddingLength = uint8(-contentLength & 7)
 | 
						|
}
 | 
						|
 | 
						|
type record struct {
 | 
						|
	h    header
 | 
						|
	rbuf []byte
 | 
						|
}
 | 
						|
 | 
						|
func (rec *record) read(r io.Reader) (buf []byte, err error) {
 | 
						|
	if err = binary.Read(r, binary.BigEndian, &rec.h); err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if rec.h.Version != 1 {
 | 
						|
		err = errors.New("fcgi: invalid header version")
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if rec.h.Type == FCGI_END_REQUEST {
 | 
						|
		err = io.EOF
 | 
						|
		return
 | 
						|
	}
 | 
						|
	n := int(rec.h.ContentLength) + int(rec.h.PaddingLength)
 | 
						|
	if len(rec.rbuf) < n {
 | 
						|
		rec.rbuf = make([]byte, n)
 | 
						|
	}
 | 
						|
	if n, err = io.ReadFull(r, rec.rbuf[:n]); err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	buf = rec.rbuf[:int(rec.h.ContentLength)]
 | 
						|
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
type FCGIClient struct {
 | 
						|
	mutex     sync.Mutex
 | 
						|
	rwc       io.ReadWriteCloser
 | 
						|
	h         header
 | 
						|
	buf       bytes.Buffer
 | 
						|
	keepAlive bool
 | 
						|
	reqId     uint16
 | 
						|
}
 | 
						|
 | 
						|
// Dial connects to the fcgi responder at the specified network address.
 | 
						|
// See func net.Dial for a description of the network and address parameters.
 | 
						|
func Dial(network, address string) (fcgi *FCGIClient, err error) {
 | 
						|
	var conn net.Conn
 | 
						|
 | 
						|
	conn, err = net.Dial(network, address)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	fcgi = &FCGIClient{
 | 
						|
		rwc:       conn,
 | 
						|
		keepAlive: false,
 | 
						|
		reqId:     1,
 | 
						|
	}
 | 
						|
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
// Close closes fcgi connnection
 | 
						|
func (c *FCGIClient) Close() {
 | 
						|
	c.rwc.Close()
 | 
						|
}
 | 
						|
 | 
						|
func (c *FCGIClient) writeRecord(recType uint8, content []byte) (err error) {
 | 
						|
	c.mutex.Lock()
 | 
						|
	defer c.mutex.Unlock()
 | 
						|
	c.buf.Reset()
 | 
						|
	c.h.init(recType, c.reqId, len(content))
 | 
						|
	if err := binary.Write(&c.buf, binary.BigEndian, c.h); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if _, err := c.buf.Write(content); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if _, err := c.buf.Write(pad[:c.h.PaddingLength]); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	_, err = c.rwc.Write(c.buf.Bytes())
 | 
						|
	return err
 | 
						|
}
 | 
						|
 | 
						|
func (c *FCGIClient) writeBeginRequest(role uint16, flags uint8) error {
 | 
						|
	b := [8]byte{byte(role >> 8), byte(role), flags}
 | 
						|
	return c.writeRecord(FCGI_BEGIN_REQUEST, b[:])
 | 
						|
}
 | 
						|
 | 
						|
func (c *FCGIClient) writeEndRequest(appStatus int, protocolStatus uint8) error {
 | 
						|
	b := make([]byte, 8)
 | 
						|
	binary.BigEndian.PutUint32(b, uint32(appStatus))
 | 
						|
	b[4] = protocolStatus
 | 
						|
	return c.writeRecord(FCGI_END_REQUEST, b)
 | 
						|
}
 | 
						|
 | 
						|
func (c *FCGIClient) writePairs(recType uint8, pairs map[string]string) error {
 | 
						|
	w := newWriter(c, recType)
 | 
						|
	b := make([]byte, 8)
 | 
						|
	nn := 0
 | 
						|
	for k, v := range pairs {
 | 
						|
		m := 8 + len(k) + len(v)
 | 
						|
		if m > maxWrite {
 | 
						|
			// param data size exceed 65535 bytes"
 | 
						|
			vl := maxWrite - 8 - len(k)
 | 
						|
			v = v[:vl]
 | 
						|
		}
 | 
						|
		n := encodeSize(b, uint32(len(k)))
 | 
						|
		n += encodeSize(b[n:], uint32(len(v)))
 | 
						|
		m = n + len(k) + len(v)
 | 
						|
		if (nn + m) > maxWrite {
 | 
						|
			w.Flush()
 | 
						|
			nn = 0
 | 
						|
		}
 | 
						|
		nn += m
 | 
						|
		if _, err := w.Write(b[:n]); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		if _, err := w.WriteString(k); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		if _, err := w.WriteString(v); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	w.Close()
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func readSize(s []byte) (uint32, int) {
 | 
						|
	if len(s) == 0 {
 | 
						|
		return 0, 0
 | 
						|
	}
 | 
						|
	size, n := uint32(s[0]), 1
 | 
						|
	if size&(1<<7) != 0 {
 | 
						|
		if len(s) < 4 {
 | 
						|
			return 0, 0
 | 
						|
		}
 | 
						|
		n = 4
 | 
						|
		size = binary.BigEndian.Uint32(s)
 | 
						|
		size &^= 1 << 31
 | 
						|
	}
 | 
						|
	return size, n
 | 
						|
}
 | 
						|
 | 
						|
func readString(s []byte, size uint32) string {
 | 
						|
	if size > uint32(len(s)) {
 | 
						|
		return ""
 | 
						|
	}
 | 
						|
	return string(s[:size])
 | 
						|
}
 | 
						|
 | 
						|
func encodeSize(b []byte, size uint32) int {
 | 
						|
	if size > 127 {
 | 
						|
		size |= 1 << 31
 | 
						|
		binary.BigEndian.PutUint32(b, size)
 | 
						|
		return 4
 | 
						|
	}
 | 
						|
	b[0] = byte(size)
 | 
						|
	return 1
 | 
						|
}
 | 
						|
 | 
						|
// bufWriter encapsulates bufio.Writer but also closes the underlying stream when
 | 
						|
// Closed.
 | 
						|
type bufWriter struct {
 | 
						|
	closer io.Closer
 | 
						|
	*bufio.Writer
 | 
						|
}
 | 
						|
 | 
						|
func (w *bufWriter) Close() error {
 | 
						|
	if err := w.Writer.Flush(); err != nil {
 | 
						|
		w.closer.Close()
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	return w.closer.Close()
 | 
						|
}
 | 
						|
 | 
						|
func newWriter(c *FCGIClient, recType uint8) *bufWriter {
 | 
						|
	s := &streamWriter{c: c, recType: recType}
 | 
						|
	w := bufio.NewWriterSize(s, maxWrite)
 | 
						|
	return &bufWriter{s, w}
 | 
						|
}
 | 
						|
 | 
						|
// streamWriter abstracts out the separation of a stream into discrete records.
 | 
						|
// It only writes maxWrite bytes at a time.
 | 
						|
type streamWriter struct {
 | 
						|
	c       *FCGIClient
 | 
						|
	recType uint8
 | 
						|
}
 | 
						|
 | 
						|
func (w *streamWriter) Write(p []byte) (int, error) {
 | 
						|
	nn := 0
 | 
						|
	for len(p) > 0 {
 | 
						|
		n := len(p)
 | 
						|
		if n > maxWrite {
 | 
						|
			n = maxWrite
 | 
						|
		}
 | 
						|
		if err := w.c.writeRecord(w.recType, p[:n]); err != nil {
 | 
						|
			return nn, err
 | 
						|
		}
 | 
						|
		nn += n
 | 
						|
		p = p[n:]
 | 
						|
	}
 | 
						|
	return nn, nil
 | 
						|
}
 | 
						|
 | 
						|
func (w *streamWriter) Close() error {
 | 
						|
	// send empty record to close the stream
 | 
						|
	return w.c.writeRecord(w.recType, nil)
 | 
						|
}
 | 
						|
 | 
						|
type streamReader struct {
 | 
						|
	c   *FCGIClient
 | 
						|
	buf []byte
 | 
						|
}
 | 
						|
 | 
						|
func (w *streamReader) Read(p []byte) (n int, err error) {
 | 
						|
 | 
						|
	if len(p) > 0 {
 | 
						|
		if len(w.buf) == 0 {
 | 
						|
			rec := &record{}
 | 
						|
			w.buf, err = rec.read(w.c.rwc)
 | 
						|
			if err != nil {
 | 
						|
				return
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		n = len(p)
 | 
						|
		if n > len(w.buf) {
 | 
						|
			n = len(w.buf)
 | 
						|
		}
 | 
						|
		copy(p, w.buf[:n])
 | 
						|
		w.buf = w.buf[n:]
 | 
						|
	}
 | 
						|
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
// Do made the request and returns a io.Reader that translates the data read
 | 
						|
// from fcgi responder out of fcgi packet before returning it.
 | 
						|
func (c *FCGIClient) Do(p map[string]string, req io.Reader) (r io.Reader, err error) {
 | 
						|
	err = c.writeBeginRequest(uint16(FCGI_RESPONDER), 0)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	err = c.writePairs(FCGI_PARAMS, p)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	body := newWriter(c, FCGI_STDIN)
 | 
						|
	if req != nil {
 | 
						|
		io.Copy(body, req)
 | 
						|
	}
 | 
						|
	body.Close()
 | 
						|
 | 
						|
	r = &streamReader{c: c}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
// Request returns a HTTP Response with Header and Body
 | 
						|
// from fcgi responder
 | 
						|
func (c *FCGIClient) Request(p map[string]string, req io.Reader) (resp *http.Response, err error) {
 | 
						|
 | 
						|
	r, err := c.Do(p, req)
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	rb := bufio.NewReader(r)
 | 
						|
	tp := textproto.NewReader(rb)
 | 
						|
	resp = new(http.Response)
 | 
						|
 | 
						|
	// Parse the response headers.
 | 
						|
	mimeHeader, err := tp.ReadMIMEHeader()
 | 
						|
	if err != nil && err != io.EOF {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	resp.Header = http.Header(mimeHeader)
 | 
						|
 | 
						|
	if resp.Header.Get("Status") != "" {
 | 
						|
		statusParts := strings.SplitN(resp.Header.Get("Status"), " ", 2)
 | 
						|
		resp.StatusCode, err = strconv.Atoi(statusParts[0])
 | 
						|
		if err != nil {
 | 
						|
			return
 | 
						|
		}
 | 
						|
		if len(statusParts) > 1 {
 | 
						|
			resp.Status = statusParts[1]	
 | 
						|
		}
 | 
						|
		
 | 
						|
	} else {
 | 
						|
		resp.StatusCode = http.StatusOK
 | 
						|
	}
 | 
						|
 | 
						|
	// TODO: fixTransferEncoding ?
 | 
						|
	resp.TransferEncoding = resp.Header["Transfer-Encoding"]
 | 
						|
	resp.ContentLength, _ = strconv.ParseInt(resp.Header.Get("Content-Length"), 10, 64)
 | 
						|
 | 
						|
	if chunked(resp.TransferEncoding) {
 | 
						|
		resp.Body = ioutil.NopCloser(httputil.NewChunkedReader(rb))
 | 
						|
	} else {
 | 
						|
		resp.Body = ioutil.NopCloser(rb)
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
// Get issues a GET request to the fcgi responder.
 | 
						|
func (c *FCGIClient) Get(p map[string]string) (resp *http.Response, err error) {
 | 
						|
 | 
						|
	p["REQUEST_METHOD"] = "GET"
 | 
						|
	p["CONTENT_LENGTH"] = "0"
 | 
						|
 | 
						|
	return c.Request(p, nil)
 | 
						|
}
 | 
						|
 | 
						|
// Head issues a HEAD request to the fcgi responder.
 | 
						|
func (c *FCGIClient) Head(p map[string]string) (resp *http.Response, err error) {
 | 
						|
 | 
						|
	p["REQUEST_METHOD"] = "HEAD"
 | 
						|
	p["CONTENT_LENGTH"] = "0"
 | 
						|
 | 
						|
	return c.Request(p, nil)
 | 
						|
}
 | 
						|
 | 
						|
// Options issues an OPTIONS request to the fcgi responder.
 | 
						|
func (c *FCGIClient) Options(p map[string]string) (resp *http.Response, err error) {
 | 
						|
 | 
						|
	p["REQUEST_METHOD"] = "OPTIONS"
 | 
						|
	p["CONTENT_LENGTH"] = "0"
 | 
						|
 | 
						|
	return c.Request(p, nil)
 | 
						|
}
 | 
						|
 | 
						|
// Post issues a POST request to the fcgi responder. with request body
 | 
						|
// in the format that bodyType specified
 | 
						|
func (c *FCGIClient) Post(p map[string]string, bodyType string, body io.Reader, l int) (resp *http.Response, err error) {
 | 
						|
 | 
						|
	if len(p["REQUEST_METHOD"]) == 0 || p["REQUEST_METHOD"] == "GET" {
 | 
						|
		p["REQUEST_METHOD"] = "POST"
 | 
						|
	}
 | 
						|
	p["CONTENT_LENGTH"] = strconv.Itoa(l)
 | 
						|
	if len(bodyType) > 0 {
 | 
						|
		p["CONTENT_TYPE"] = bodyType
 | 
						|
	} else {
 | 
						|
		p["CONTENT_TYPE"] = "application/x-www-form-urlencoded"
 | 
						|
	}
 | 
						|
 | 
						|
	return c.Request(p, body)
 | 
						|
}
 | 
						|
 | 
						|
// Put issues a PUT request to the fcgi responder.
 | 
						|
func (c *FCGIClient) Put(p map[string]string, bodyType string, body io.Reader, l int) (resp *http.Response, err error) {
 | 
						|
 | 
						|
	p["REQUEST_METHOD"] = "PUT"
 | 
						|
 | 
						|
	return c.Post(p, bodyType, body, l)
 | 
						|
}
 | 
						|
 | 
						|
// Patch issues a PATCH request to the fcgi responder.
 | 
						|
func (c *FCGIClient) Patch(p map[string]string, bodyType string, body io.Reader, l int) (resp *http.Response, err error) {
 | 
						|
 | 
						|
	p["REQUEST_METHOD"] = "PATCH"
 | 
						|
 | 
						|
	return c.Post(p, bodyType, body, l)
 | 
						|
}
 | 
						|
 | 
						|
// Delete issues a DELETE request to the fcgi responder.
 | 
						|
func (c *FCGIClient) Delete(p map[string]string, bodyType string, body io.Reader, l int) (resp *http.Response, err error) {
 | 
						|
 | 
						|
	p["REQUEST_METHOD"] = "DELETE"
 | 
						|
 | 
						|
	return c.Post(p, bodyType, body, l)
 | 
						|
}
 | 
						|
 | 
						|
// PostForm issues a POST to the fcgi responder, with form
 | 
						|
// as a string key to a list values (url.Values)
 | 
						|
func (c *FCGIClient) PostForm(p map[string]string, data url.Values) (resp *http.Response, err error) {
 | 
						|
	body := bytes.NewReader([]byte(data.Encode()))
 | 
						|
	return c.Post(p, "application/x-www-form-urlencoded", body, body.Len())
 | 
						|
}
 | 
						|
 | 
						|
// PostFile issues a POST to the fcgi responder in multipart(RFC 2046) standard,
 | 
						|
// with form as a string key to a list values (url.Values),
 | 
						|
// and/or with file as a string key to a list file path.
 | 
						|
func (c *FCGIClient) PostFile(p map[string]string, data url.Values, file map[string]string) (resp *http.Response, err error) {
 | 
						|
	buf := &bytes.Buffer{}
 | 
						|
	writer := multipart.NewWriter(buf)
 | 
						|
	bodyType := writer.FormDataContentType()
 | 
						|
 | 
						|
	for key, val := range data {
 | 
						|
		for _, v0 := range val {
 | 
						|
			err = writer.WriteField(key, v0)
 | 
						|
			if err != nil {
 | 
						|
				return
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	for key, val := range file {
 | 
						|
		fd, e := os.Open(val)
 | 
						|
		if e != nil {
 | 
						|
			return nil, e
 | 
						|
		}
 | 
						|
		defer fd.Close()
 | 
						|
 | 
						|
		part, e := writer.CreateFormFile(key, filepath.Base(val))
 | 
						|
		if e != nil {
 | 
						|
			return nil, e
 | 
						|
		}
 | 
						|
		_, err = io.Copy(part, fd)
 | 
						|
	}
 | 
						|
 | 
						|
	err = writer.Close()
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	return c.Post(p, bodyType, buf, buf.Len())
 | 
						|
}
 | 
						|
 | 
						|
// Checks whether chunked is part of the encodings stack
 | 
						|
func chunked(te []string) bool { return len(te) > 0 && te[0] == "chunked" }
 |