mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-31 10:37:24 -04:00 
			
		
		
		
	* WIP * HTTP2/Push for golang 1.8 * Push plugin completed for review * Correct build tag * Move push plugin position * Add build tags to tests * Gofmt that code * Add header/method validations * Load push plugin * Fixes for wrapping writers * Push after delivering file * Fixes, review changes * Remove build tags, support new syntax * Fix spelling * gofmt -s -w . * Gogland time * Add interface guards * gofmt * After review fixes
		
			
				
	
	
		
			173 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			173 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package push
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"net/http"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/mholt/caddy"
 | |
| 	"github.com/mholt/caddy/caddyhttp/httpserver"
 | |
| )
 | |
| 
 | |
| func init() {
 | |
| 	caddy.RegisterPlugin("push", caddy.Plugin{
 | |
| 		ServerType: "http",
 | |
| 		Action:     setup,
 | |
| 	})
 | |
| }
 | |
| 
 | |
| var errInvalidHeader = errors.New("header directive requires [name] [value]")
 | |
| 
 | |
| var errHeaderStartsWithColon = errors.New("header cannot start with colon")
 | |
| var errMethodNotSupported = errors.New("push supports only GET and HEAD methods")
 | |
| 
 | |
| const pushHeader = "X-Push"
 | |
| 
 | |
| var emptyRules = []Rule{}
 | |
| 
 | |
| // setup configures a new Push middleware
 | |
| func setup(c *caddy.Controller) error {
 | |
| 	rules, err := parsePushRules(c)
 | |
| 
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	httpserver.GetConfig(c).AddMiddleware(func(next httpserver.Handler) httpserver.Handler {
 | |
| 		return Middleware{Next: next, Rules: rules}
 | |
| 	})
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func parsePushRules(c *caddy.Controller) ([]Rule, error) {
 | |
| 	var rules = make(map[string]*Rule)
 | |
| 
 | |
| 	for c.NextLine() {
 | |
| 		if !c.NextArg() {
 | |
| 			return emptyRules, c.ArgErr()
 | |
| 		}
 | |
| 
 | |
| 		path := c.Val()
 | |
| 		args := c.RemainingArgs()
 | |
| 
 | |
| 		var rule *Rule
 | |
| 		var resources []Resource
 | |
| 		var ops []ruleOp
 | |
| 
 | |
| 		if existingRule, ok := rules[path]; ok {
 | |
| 			rule = existingRule
 | |
| 		} else {
 | |
| 			rule = new(Rule)
 | |
| 			rule.Path = path
 | |
| 			rules[rule.Path] = rule
 | |
| 		}
 | |
| 
 | |
| 		for i := 0; i < len(args); i++ {
 | |
| 			resources = append(resources, Resource{
 | |
| 				Path:   args[i],
 | |
| 				Method: http.MethodGet,
 | |
| 				Header: http.Header{pushHeader: []string{}},
 | |
| 			})
 | |
| 		}
 | |
| 
 | |
| 		for c.NextBlock() {
 | |
| 			val := c.Val()
 | |
| 
 | |
| 			switch val {
 | |
| 			case "method":
 | |
| 				if !c.NextArg() {
 | |
| 					return emptyRules, c.ArgErr()
 | |
| 				}
 | |
| 
 | |
| 				method := c.Val()
 | |
| 
 | |
| 				if err := validateMethod(method); err != nil {
 | |
| 					return emptyRules, errMethodNotSupported
 | |
| 				}
 | |
| 
 | |
| 				ops = append(ops, setMethodOp(method))
 | |
| 
 | |
| 			case "header":
 | |
| 				args := c.RemainingArgs()
 | |
| 
 | |
| 				if len(args) != 2 {
 | |
| 					return emptyRules, errInvalidHeader
 | |
| 				}
 | |
| 
 | |
| 				if err := validateHeader(args[0]); err != nil {
 | |
| 					return emptyRules, err
 | |
| 				}
 | |
| 
 | |
| 				ops = append(ops, setHeaderOp(args[0], args[1]))
 | |
| 
 | |
| 			default:
 | |
| 				resources = append(resources, Resource{
 | |
| 					Path:   val,
 | |
| 					Method: http.MethodGet,
 | |
| 					Header: http.Header{pushHeader: []string{}},
 | |
| 				})
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		for _, op := range ops {
 | |
| 			op(resources)
 | |
| 		}
 | |
| 
 | |
| 		rule.Resources = append(rule.Resources, resources...)
 | |
| 	}
 | |
| 
 | |
| 	var returnRules []Rule
 | |
| 
 | |
| 	for path, rule := range rules {
 | |
| 		if len(rule.Resources) == 0 {
 | |
| 			return emptyRules, c.Errf("Rule %s has empty push resources list", path)
 | |
| 		}
 | |
| 
 | |
| 		returnRules = append(returnRules, *rule)
 | |
| 	}
 | |
| 
 | |
| 	return returnRules, nil
 | |
| }
 | |
| 
 | |
| func setHeaderOp(key, value string) func(resources []Resource) {
 | |
| 	return func(resources []Resource) {
 | |
| 		for index := range resources {
 | |
| 			resources[index].Header.Set(key, value)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func setMethodOp(method string) func(resources []Resource) {
 | |
| 
 | |
| 	return func(resources []Resource) {
 | |
| 		for index := range resources {
 | |
| 			resources[index].Method = method
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func validateHeader(header string) error {
 | |
| 	if strings.HasPrefix(header, ":") {
 | |
| 		return errHeaderStartsWithColon
 | |
| 	}
 | |
| 
 | |
| 	switch strings.ToLower(header) {
 | |
| 	case "content-length", "content-encoding", "trailer", "te", "expect", "host":
 | |
| 		return fmt.Errorf("push headers cannot include %s", header)
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // rules based on https://go-review.googlesource.com/#/c/29439/4/http2/go18.go#94
 | |
| func validateMethod(method string) error {
 | |
| 	if method != http.MethodGet && method != http.MethodHead {
 | |
| 		return errMethodNotSupported
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 |