mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-25 07:49:19 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			517 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			517 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
 | |
| }
 | |
| 
 | |
| // 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 fcgi connnection
 | |
| func (this *FCGIClient) Close() {
 | |
| 	this.rwc.Close()
 | |
| }
 | |
| 
 | |
| func (this *FCGIClient) writeRecord(recType uint8, content []byte) (err error) {
 | |
| 	this.mutex.Lock()
 | |
| 	defer this.mutex.Unlock()
 | |
| 	this.buf.Reset()
 | |
| 	this.h.init(recType, this.reqId, len(content))
 | |
| 	if err := binary.Write(&this.buf, binary.BigEndian, this.h); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	if _, err := this.buf.Write(content); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	if _, err := this.buf.Write(pad[:this.h.PaddingLength]); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	_, err = this.rwc.Write(this.buf.Bytes())
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (this *FCGIClient) writeBeginRequest(role uint16, flags uint8) error {
 | |
| 	b := [8]byte{byte(role >> 8), byte(role), flags}
 | |
| 	return this.writeRecord(FCGI_BEGIN_REQUEST, b[:])
 | |
| }
 | |
| 
 | |
| func (this *FCGIClient) writeEndRequest(appStatus int, protocolStatus uint8) error {
 | |
| 	b := make([]byte, 8)
 | |
| 	binary.BigEndian.PutUint32(b, uint32(appStatus))
 | |
| 	b[4] = protocolStatus
 | |
| 	return this.writeRecord(FCGI_END_REQUEST, b)
 | |
| }
 | |
| 
 | |
| func (this *FCGIClient) writePairs(recType uint8, pairs map[string]string) error {
 | |
| 	w := newWriter(this, 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 (this *FCGIClient) Do(p map[string]string, req io.Reader) (r io.Reader, err error) {
 | |
| 	err = this.writeBeginRequest(uint16(FCGI_RESPONDER), 0)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	err = this.writePairs(FCGI_PARAMS, p)
 | |
| 	if err != nil {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	body := newWriter(this, FCGI_STDIN)
 | |
| 	if req != nil {
 | |
| 		io.Copy(body, req)
 | |
| 	}
 | |
| 	body.Close()
 | |
| 
 | |
| 	r = &streamReader{c: this}
 | |
| 	return
 | |
| }
 | |
| 
 | |
| // Request returns a HTTP Response with Header and Body
 | |
| // from fcgi responder
 | |
| func (this *FCGIClient) Request(p map[string]string, req io.Reader) (resp *http.Response, err error) {
 | |
| 
 | |
| 	r, err := this.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
 | |
| 		}
 | |
| 		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 (this *FCGIClient) Get(p map[string]string) (resp *http.Response, err error) {
 | |
| 
 | |
| 	p["REQUEST_METHOD"] = "GET"
 | |
| 	p["CONTENT_LENGTH"] = "0"
 | |
| 
 | |
| 	return this.Request(p, nil)
 | |
| }
 | |
| 
 | |
| // Head issues a HEAD request to the fcgi responder.
 | |
| func (this *FCGIClient) Head(p map[string]string) (resp *http.Response, err error) {
 | |
| 
 | |
| 	p["REQUEST_METHOD"] = "HEAD"
 | |
| 	p["CONTENT_LENGTH"] = "0"
 | |
| 
 | |
| 	return this.Request(p, nil)
 | |
| }
 | |
| 
 | |
| // Options issues an OPTIONS request to the fcgi responder.
 | |
| func (this *FCGIClient) Options(p map[string]string) (resp *http.Response, err error) {
 | |
| 
 | |
| 	p["REQUEST_METHOD"] = "OPTIONS"
 | |
| 	p["CONTENT_LENGTH"] = "0"
 | |
| 
 | |
| 	return this.Request(p, nil)
 | |
| }
 | |
| 
 | |
| // Post issues a POST request to the fcgi responder. with request body
 | |
| // in the format that bodyType specified
 | |
| func (this *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 this.Request(p, body)
 | |
| }
 | |
| 
 | |
| // Put issues a PUT request to the fcgi responder.
 | |
| func (this *FCGIClient) Put(p map[string]string, bodyType string, body io.Reader, l int) (resp *http.Response, err error) {
 | |
| 
 | |
| 	p["REQUEST_METHOD"] = "PUT"
 | |
| 
 | |
| 	return this.Post(p, bodyType, body, l)
 | |
| }
 | |
| 
 | |
| // Patch issues a PATCH request to the fcgi responder.
 | |
| func (this *FCGIClient) Patch(p map[string]string, bodyType string, body io.Reader, l int) (resp *http.Response, err error) {
 | |
| 
 | |
| 	p["REQUEST_METHOD"] = "PATCH"
 | |
| 
 | |
| 	return this.Post(p, bodyType, body, l)
 | |
| }
 | |
| 
 | |
| // Delete issues a DELETE request to the fcgi responder.
 | |
| func (this *FCGIClient) Delete(p map[string]string, bodyType string, body io.Reader, l int) (resp *http.Response, err error) {
 | |
| 
 | |
| 	p["REQUEST_METHOD"] = "DELETE"
 | |
| 
 | |
| 	return this.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 (this *FCGIClient) PostForm(p map[string]string, data url.Values) (resp *http.Response, err error) {
 | |
| 	body := bytes.NewReader([]byte(data.Encode()))
 | |
| 	return this.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 (this *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 this.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" }
 |