mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-25 15:52:45 -04:00 
			
		
		
		
	generalizing fastcgi parameters, and improving headers passed.
This commit is contained in:
		
							parent
							
								
									9e12c45d82
								
							
						
					
					
						commit
						1ac32a5256
					
				| @ -4,6 +4,8 @@ | |||||||
| package fastcgi | package fastcgi | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"errors" | ||||||
|  | 	"fmt" | ||||||
| 	"io" | 	"io" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| @ -17,7 +19,10 @@ import ( | |||||||
| 
 | 
 | ||||||
| // New generates a new FastCGI middleware. | // New generates a new FastCGI middleware. | ||||||
| func New(c middleware.Controller) (middleware.Middleware, error) { | func New(c middleware.Controller) (middleware.Middleware, error) { | ||||||
| 	root := c.Root() | 	root, err := filepath.Abs(c.Root()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	rules, err := parse(c) | 	rules, err := parse(c) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -25,130 +30,180 @@ func New(c middleware.Controller) (middleware.Middleware, error) { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return func(next middleware.Handler) middleware.Handler { | 	return func(next middleware.Handler) middleware.Handler { | ||||||
| 		return Handler{Next: next, Rules: rules, Root: root} | 		return Handler{ | ||||||
|  | 			Next:            next, | ||||||
|  | 			Rules:           rules, | ||||||
|  | 			Root:            root, | ||||||
|  | 			SoftwareName:    "Caddy", // TODO: Once generators are not in the same pkg as handler, obtain this from some global const | ||||||
|  | 			SoftwareVersion: "",      // TODO: Get this from some global const too | ||||||
|  | 			// TODO: Set ServerName and ServerPort to correct values... (as user defined in config) | ||||||
|  | 		} | ||||||
| 	}, nil | 	}, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Handler is a middleware type that can handle requests as a FastCGI client. | // Handler is a middleware type that can handle requests as a FastCGI client. | ||||||
| type Handler struct { | type Handler struct { | ||||||
| 	Next  middleware.Handler | 	Next  middleware.Handler | ||||||
| 	Root  string | 	Root  string // must be absolute path to site root | ||||||
| 	Rules []Rule | 	Rules []Rule | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| func (h Handler) DoesFileExist(path string) bool { | 	// These are sent to CGI scripts in env variables | ||||||
| 	file := h.Root + path | 	SoftwareName    string | ||||||
| 	if _, err := os.Stat(file); err == nil { | 	SoftwareVersion string | ||||||
| 		return true | 	ServerName      string | ||||||
| 	} | 	ServerPort      string | ||||||
| 	return false |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // ServeHTTP satisfies the middleware.Handler interface. | // ServeHTTP satisfies the middleware.Handler interface. | ||||||
| func (h Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) { | func (h Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) { | ||||||
| 	servedFcgi := false |  | ||||||
| 	indexFile := "index.php" |  | ||||||
| 	ext := ".php" |  | ||||||
| 	splitText := ".php" |  | ||||||
| 	for _, rule := range h.Rules { | 	for _, rule := range h.Rules { | ||||||
| 		if middleware.Path(r.URL.Path).Matches(rule.Path) && (strings.HasSuffix(r.URL.Path, "/") || | 		// In addition to matching the path, a request must meet some | ||||||
| 			strings.HasSuffix(r.URL.Path, ext) || !h.DoesFileExist(r.URL.Path)) { | 		// other criteria before being proxied as FastCGI. For example, | ||||||
|  | 		// we probably want to exclude static assets (CSS, JS, images...) | ||||||
|  | 		// but we also want to be flexible for the script we proxy to. | ||||||
| 
 | 
 | ||||||
| 			// Get absolute file paths | 		// These criteria work well in this order for PHP sites | ||||||
| 			absPath, err := filepath.Abs(h.Root + r.URL.Path) | 		if middleware.Path(r.URL.Path).Matches(rule.Path) && | ||||||
|  | 			(r.URL.Path[len(r.URL.Path)-1] == '/' || | ||||||
|  | 				strings.HasSuffix(r.URL.Path, rule.Ext) || | ||||||
|  | 				!h.exists(r.URL.Path)) { | ||||||
|  | 
 | ||||||
|  | 			// Create environment for CGI script | ||||||
|  | 			env, err := h.buildEnv(r, rule) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return http.StatusInternalServerError, err | 				return http.StatusInternalServerError, err | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			// Get absolute file path to website root | 			// Connect to FastCGI gateway | ||||||
| 			absRootPath, err := filepath.Abs(h.Root) |  | ||||||
| 			if err != nil { |  | ||||||
| 				return http.StatusInternalServerError, err |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			// Separate remote IP and port |  | ||||||
| 			var ip, port string |  | ||||||
| 			if idx := strings.Index(r.RemoteAddr, ":"); idx > -1 { |  | ||||||
| 				ip = r.RemoteAddr[:idx] |  | ||||||
| 				port = r.RemoteAddr[idx:] |  | ||||||
| 			} else { |  | ||||||
| 				ip = r.RemoteAddr |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			// TODO: Do we really have to make this map from scratch for each request? |  | ||||||
| 			// TODO: We have quite a few more to map, too. |  | ||||||
| 			env := make(map[string]string) |  | ||||||
| 			env["SERVER_NAME"] = "caddy" |  | ||||||
| 			env["SERVER_SOFTWARE"] = "caddy" // TODO: Obtain version info... |  | ||||||
| 			env["SERVER_PROTOCOL"] = r.Proto |  | ||||||
| 			env["SCRIPT_FILENAME"] = absPath |  | ||||||
| 			env["REMOTE_ADDR"] = ip |  | ||||||
| 			env["REMOTE_PORT"] = port |  | ||||||
| 			env["REQUEST_METHOD"] = r.Method |  | ||||||
| 			env["QUERY_STRING"] = r.URL.RawQuery |  | ||||||
| 			env["SCRIPT_NAME"] = r.URL.Path |  | ||||||
| 			env["HTTP_HOST"] = r.Host |  | ||||||
| 
 |  | ||||||
| 			split := strings.Index(r.URL.Path, splitText) |  | ||||||
| 
 |  | ||||||
| 			if split == -1 { |  | ||||||
| 				//request doesn't have the extension |  | ||||||
| 				//send the request to the index file |  | ||||||
| 				env["DOCUMENT_URI"] = "/" + indexFile |  | ||||||
| 				env["SCRIPT_NAME"] = "/" + indexFile |  | ||||||
| 				env["SCRIPT_FILENAME"] = absRootPath + "/" + indexFile |  | ||||||
| 				env["PATH_INFO"] = r.URL.Path |  | ||||||
| 			} else { |  | ||||||
| 				env["DOCUMENT_URI"] = r.URL.Path[:split+len(splitText)] |  | ||||||
| 				env["PATH_INFO"] = r.URL.Path[split+len(splitText):] |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			env["REQUEST_URI"] = r.URL.RequestURI() |  | ||||||
| 
 |  | ||||||
| 			env["DOCUMENT_ROOT"] = absRootPath |  | ||||||
| 			env["HTTP_COOKIE"] = r.Header.Get("Cookie") |  | ||||||
| 
 |  | ||||||
| 			fcgi, err := Dial("tcp", rule.Address) | 			fcgi, err := Dial("tcp", rule.Address) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return http.StatusBadGateway, err | 				return http.StatusBadGateway, err | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
|  | 			// TODO: Allow more methods (requires refactoring fcgiclient first...) | ||||||
| 			var resp *http.Response | 			var resp *http.Response | ||||||
| 			if r.Method == "GET" { | 			switch r.Method { | ||||||
|  | 			case "GET": | ||||||
| 				resp, err = fcgi.Get(env) | 				resp, err = fcgi.Get(env) | ||||||
| 			} else { | 			case "POST": | ||||||
| 				l, _ := strconv.Atoi(r.Header.Get("Content-Length")) | 				l, _ := strconv.Atoi(r.Header.Get("Content-Length")) | ||||||
| 				resp, err = fcgi.Post(env, r.Header.Get("Content-Type"), r.Body, l) | 				resp, err = fcgi.Post(env, r.Header.Get("Content-Type"), r.Body, l) | ||||||
|  | 			default: | ||||||
|  | 				return http.StatusMethodNotAllowed, nil | ||||||
| 			} | 			} | ||||||
|  | 			defer resp.Body.Close() | ||||||
|  | 
 | ||||||
| 			if err != nil && err != io.EOF { | 			if err != nil && err != io.EOF { | ||||||
| 				return http.StatusBadGateway, err | 				return http.StatusBadGateway, err | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			body, err := ioutil.ReadAll(resp.Body) | 			// Write the response header | ||||||
| 			if err != nil { |  | ||||||
| 				return http.StatusBadGateway, err |  | ||||||
| 			} |  | ||||||
| 
 |  | ||||||
| 			for key, vals := range resp.Header { | 			for key, vals := range resp.Header { | ||||||
| 				for _, val := range vals { | 				for _, val := range vals { | ||||||
| 					w.Header().Add(key, val) | 					w.Header().Add(key, val) | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 
 |  | ||||||
| 			w.WriteHeader(resp.StatusCode) | 			w.WriteHeader(resp.StatusCode) | ||||||
| 			w.Write(body) |  | ||||||
| 
 | 
 | ||||||
| 			servedFcgi = true | 			body, err := ioutil.ReadAll(resp.Body) | ||||||
|  | 			fmt.Printf("%s", body) | ||||||
|  | 			fmt.Printf("%d\n", resp.StatusCode) | ||||||
|  | 			fmt.Printf("%d\n", len(body)) | ||||||
|  | 			w.Write(body) | ||||||
| 
 | 
 | ||||||
| 			return resp.StatusCode, nil | 			return resp.StatusCode, nil | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if !servedFcgi { |  | ||||||
| 	return h.Next.ServeHTTP(w, r) | 	return h.Next.ServeHTTP(w, r) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 	return 0, nil | func (h Handler) exists(path string) bool { | ||||||
|  | 	if _, err := os.Stat(h.Root + path); err == nil { | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (h Handler) buildEnv(r *http.Request, rule Rule) (map[string]string, error) { | ||||||
|  | 	var env map[string]string | ||||||
|  | 
 | ||||||
|  | 	// Get absolute path of requested resource | ||||||
|  | 	absPath, err := filepath.Abs(h.Root + r.URL.Path) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return env, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Separate remote IP and port; more lenient than net.SplitHostPort | ||||||
|  | 	var ip, port string | ||||||
|  | 	if idx := strings.Index(r.RemoteAddr, ":"); idx > -1 { | ||||||
|  | 		ip = r.RemoteAddr[:idx] | ||||||
|  | 		port = r.RemoteAddr[idx+1:] | ||||||
|  | 	} else { | ||||||
|  | 		ip = r.RemoteAddr | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Split path in preparation for env variables | ||||||
|  | 	splitPos := strings.Index(r.URL.Path, rule.SplitPath) | ||||||
|  | 	var docURI, scriptName, scriptFilename, pathInfo string | ||||||
|  | 	if splitPos == -1 { | ||||||
|  | 		// Request doesn't have the extension, so assume index file | ||||||
|  | 		docURI = "/" + rule.IndexFile | ||||||
|  | 		scriptName = "/" + rule.IndexFile | ||||||
|  | 		scriptFilename = h.Root + "/" + rule.IndexFile | ||||||
|  | 		pathInfo = r.URL.Path | ||||||
|  | 	} else { | ||||||
|  | 		// Request has the extension; path was split successfully | ||||||
|  | 		docURI = r.URL.Path[:splitPos+len(rule.SplitPath)] | ||||||
|  | 		pathInfo = r.URL.Path[splitPos+len(rule.SplitPath):] | ||||||
|  | 		scriptName = r.URL.Path | ||||||
|  | 		scriptFilename = absPath | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Some variables are unused but cleared explicitly to prevent | ||||||
|  | 	// the parent environment from interfering. | ||||||
|  | 	env = map[string]string{ | ||||||
|  | 
 | ||||||
|  | 		// Variables defined in CGI 1.1 spec | ||||||
|  | 		"AUTH_TYPE":         "", // Not used | ||||||
|  | 		"CONTENT_LENGTH":    r.Header.Get("Content-Length"), | ||||||
|  | 		"CONTENT_TYPE":      r.Header.Get("Content-Type"), | ||||||
|  | 		"GATEWAY_INTERFACE": "CGI/1.1", | ||||||
|  | 		"PATH_INFO":         pathInfo, | ||||||
|  | 		"PATH_TRANSLATED":   h.Root + "/" + pathInfo, // Source for path_translated: http://www.oreilly.com/openbook/cgi/ch02_04.html | ||||||
|  | 		"QUERY_STRING":      r.URL.RawQuery, | ||||||
|  | 		"REMOTE_ADDR":       ip, | ||||||
|  | 		"REMOTE_HOST":       ip, // For speed, remote host lookups disabled | ||||||
|  | 		"REMOTE_PORT":       port, | ||||||
|  | 		"REMOTE_IDENT":      "", // Not used | ||||||
|  | 		"REMOTE_USER":       "", // Not used | ||||||
|  | 		"REQUEST_METHOD":    r.Method, | ||||||
|  | 		"SERVER_NAME":       h.ServerName, | ||||||
|  | 		"SERVER_PORT":       h.ServerPort, | ||||||
|  | 		"SERVER_PROTOCOL":   r.Proto, | ||||||
|  | 		"SERVER_SOFTWARE":   h.SoftwareName + "/" + h.SoftwareVersion, | ||||||
|  | 
 | ||||||
|  | 		// Other variables | ||||||
|  | 		"DOCUMENT_ROOT":   h.Root, | ||||||
|  | 		"DOCUMENT_URI":    docURI, | ||||||
|  | 		"HTTP_HOST":       r.Host, // added here, since not always part of headers | ||||||
|  | 		"REQUEST_URI":     r.URL.RequestURI(), | ||||||
|  | 		"SCRIPT_FILENAME": scriptFilename, | ||||||
|  | 		"SCRIPT_NAME":     scriptName, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Add all HTTP headers to env variables | ||||||
|  | 	for field, val := range r.Header { | ||||||
|  | 		header := strings.ToUpper(field) | ||||||
|  | 		header = headerNameReplacer.Replace(header) | ||||||
|  | 		// We don't want to pass the encoding header to prevent the fastcgi server from gzipping | ||||||
|  | 		// TODO: is there a better way. | ||||||
|  | 		if header != "ACCEPT_ENCODING" { | ||||||
|  | 			env["HTTP_"+header] = strings.Join(val, ", ") | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return env, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func parse(c middleware.Controller) ([]Rule, error) { | func parse(c middleware.Controller) ([]Rule, error) { | ||||||
| @ -156,16 +211,86 @@ func parse(c middleware.Controller) ([]Rule, error) { | |||||||
| 
 | 
 | ||||||
| 	for c.Next() { | 	for c.Next() { | ||||||
| 		var rule Rule | 		var rule Rule | ||||||
| 		if !c.Args(&rule.Path, &rule.Address) { | 
 | ||||||
|  | 		args := c.RemainingArgs() | ||||||
|  | 
 | ||||||
|  | 		switch len(args) { | ||||||
|  | 		case 0: | ||||||
|  | 			return rules, c.ArgErr() | ||||||
|  | 		case 1: | ||||||
|  | 			rule.Path = "/" | ||||||
|  | 			rule.Address = args[0] | ||||||
|  | 		case 2: | ||||||
|  | 			rule.Path = args[0] | ||||||
|  | 			rule.Address = args[1] | ||||||
|  | 		case 3: | ||||||
|  | 			rule.Path = args[0] | ||||||
|  | 			rule.Address = args[1] | ||||||
|  | 			err := preset(args[2], &rule) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return rules, c.Err("Invalid fastcgi rule preset '" + args[2] + "'") | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		for c.NextBlock() { | ||||||
|  | 			switch c.Val() { | ||||||
|  | 			case "ext": | ||||||
|  | 				if !c.NextArg() { | ||||||
| 					return rules, c.ArgErr() | 					return rules, c.ArgErr() | ||||||
| 				} | 				} | ||||||
|  | 				rule.Ext = c.Val() | ||||||
|  | 			case "split": | ||||||
|  | 				if !c.NextArg() { | ||||||
|  | 					return rules, c.ArgErr() | ||||||
|  | 				} | ||||||
|  | 				rule.SplitPath = c.Val() | ||||||
|  | 			case "index": | ||||||
|  | 				if !c.NextArg() { | ||||||
|  | 					return rules, c.ArgErr() | ||||||
|  | 				} | ||||||
|  | 				rule.IndexFile = c.Val() | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		rules = append(rules, rule) | 		rules = append(rules, rule) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return rules, nil | 	return rules, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // preset configures rule according to name. It returns an error if | ||||||
|  | // name is not a recognized preset name. | ||||||
|  | func preset(name string, rule *Rule) error { | ||||||
|  | 	switch name { | ||||||
|  | 	case "php": | ||||||
|  | 		rule.Ext = ".php" | ||||||
|  | 		rule.SplitPath = ".php" | ||||||
|  | 		rule.IndexFile = "index.php" | ||||||
|  | 	default: | ||||||
|  | 		return errors.New(name + " is not a valid preset name") | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // Rule represents a FastCGI handling rule. | // Rule represents a FastCGI handling rule. | ||||||
| type Rule struct { | type Rule struct { | ||||||
| 	Path, Address string | 	// The base path to match. Required. | ||||||
|  | 	Path string | ||||||
|  | 
 | ||||||
|  | 	// The address of the FastCGI server. Required. | ||||||
|  | 	Address string | ||||||
|  | 
 | ||||||
|  | 	// Always process files with this extension with fastcgi. | ||||||
|  | 	Ext string | ||||||
|  | 
 | ||||||
|  | 	// The path in the URL will be split into two, with the first piece ending | ||||||
|  | 	// with the value of SplitPath. The first piece will be assumed as the | ||||||
|  | 	// actual resource (CGI script) name, and the second piece will be set to | ||||||
|  | 	// PATH_INFO for the CGI script to use. | ||||||
|  | 	SplitPath string | ||||||
|  | 
 | ||||||
|  | 	// If the URL does not indicate a file, an index file with this name will be assumed. | ||||||
|  | 	IndexFile string | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | var headerNameReplacer = strings.NewReplacer(" ", "_", "-", "_") | ||||||
|  | |||||||
| @ -376,10 +376,13 @@ func (this *FCGIClient) Request(p map[string]string, req io.Reader) (resp *http. | |||||||
| 
 | 
 | ||||||
| 	if resp.Header.Get("Status") != "" { | 	if resp.Header.Get("Status") != "" { | ||||||
| 		statusParts := strings.SplitN(resp.Header.Get("Status"), " ", 2) | 		statusParts := strings.SplitN(resp.Header.Get("Status"), " ", 2) | ||||||
| 		resp.StatusCode, _ = strconv.Atoi(statusParts[0]) | 		resp.StatusCode, err = strconv.Atoi(statusParts[0]) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
| 		resp.Status = statusParts[1] | 		resp.Status = statusParts[1] | ||||||
| 	} else { | 	} else { | ||||||
| 		resp.StatusCode = 200 | 		resp.StatusCode = http.StatusOK | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	// TODO: fixTransferEncoding ? | 	// TODO: fixTransferEncoding ? | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user