| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | // Copyright 2015 Matthew Holt and The Caddy Authors | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  | // you may not use this file except in compliance with the License. | 
					
						
							|  |  |  | // You may obtain a copy of the License at | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | //     http://www.apache.org/licenses/LICENSE-2.0 | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  | // distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  | // See the License for the specific language governing permissions and | 
					
						
							|  |  |  | // limitations under the License. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package encode | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2021-03-30 02:47:19 +02:00
										 |  |  | 	"net/http" | 
					
						
							|  |  |  | 	"strconv" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-09 12:19:56 -06:00
										 |  |  | 	"github.com/caddyserver/caddy/v2" | 
					
						
							|  |  |  | 	"github.com/caddyserver/caddy/v2/caddyconfig" | 
					
						
							|  |  |  | 	"github.com/caddyserver/caddy/v2/caddyconfig/caddyfile" | 
					
						
							|  |  |  | 	"github.com/caddyserver/caddy/v2/caddyconfig/httpcaddyfile" | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 	"github.com/caddyserver/caddy/v2/modules/caddyhttp" | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | func init() { | 
					
						
							|  |  |  | 	httpcaddyfile.RegisterHandlerDirective("encode", parseCaddyfile) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func parseCaddyfile(h httpcaddyfile.Helper) (caddyhttp.MiddlewareHandler, error) { | 
					
						
							|  |  |  | 	enc := new(Encode) | 
					
						
							|  |  |  | 	err := enc.UnmarshalCaddyfile(h.Dispenser) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return enc, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | // UnmarshalCaddyfile sets up the handler from Caddyfile tokens. Syntax: | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | //     encode [<matcher>] <formats...> { | 
					
						
							| 
									
										
										
										
											2021-03-30 02:47:19 +02:00
										 |  |  | //         gzip           [<level>] | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | //         zstd | 
					
						
							| 
									
										
										
										
											2021-03-30 02:47:19 +02:00
										 |  |  | //         minimum_length <length> | 
					
						
							|  |  |  | //         prefer         <formats...> | 
					
						
							|  |  |  | //         # response matcher block | 
					
						
							|  |  |  | //         match { | 
					
						
							|  |  |  | //             status <code...> | 
					
						
							|  |  |  | //             header <field> [<value>] | 
					
						
							|  |  |  | //         } | 
					
						
							|  |  |  | //         # or response matcher single line syntax | 
					
						
							|  |  |  | //         match [header <field> [<value>]] | [status <code...>] | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | //     } | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Specifying the formats on the first line will use those formats' defaults. | 
					
						
							|  |  |  | func (enc *Encode) UnmarshalCaddyfile(d *caddyfile.Dispenser) error { | 
					
						
							| 
									
										
										
										
											2021-03-30 02:47:19 +02:00
										 |  |  | 	responseMatchers := make(map[string]caddyhttp.ResponseMatcher) | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 	for d.Next() { | 
					
						
							|  |  |  | 		for _, arg := range d.RemainingArgs() { | 
					
						
							|  |  |  | 			mod, err := caddy.GetModule("http.encoders." + arg) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2021-03-30 02:47:19 +02:00
										 |  |  | 				return d.Errf("finding encoder module '%s': %v", mod, err) | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			encoding, ok := mod.New().(Encoding) | 
					
						
							|  |  |  | 			if !ok { | 
					
						
							| 
									
										
										
										
											2021-03-30 02:47:19 +02:00
										 |  |  | 				return d.Errf("module %s is not an HTTP encoding", mod) | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if enc.EncodingsRaw == nil { | 
					
						
							| 
									
										
										
										
											2019-12-10 13:36:46 -07:00
										 |  |  | 				enc.EncodingsRaw = make(caddy.ModuleMap) | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			enc.EncodingsRaw[arg] = caddyconfig.JSON(encoding, nil) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 19:21:52 -06:00
										 |  |  | 		for d.NextBlock(0) { | 
					
						
							| 
									
										
										
										
											2021-03-30 02:47:19 +02:00
										 |  |  | 			switch d.Val() { | 
					
						
							|  |  |  | 			case "minimum_length": | 
					
						
							|  |  |  | 				if !d.NextArg() { | 
					
						
							|  |  |  | 					return d.ArgErr() | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				minLength, err := strconv.Atoi(d.Val()) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				enc.MinLength = minLength | 
					
						
							|  |  |  | 			case "prefer": | 
					
						
							|  |  |  | 				var encs []string | 
					
						
							|  |  |  | 				for d.NextArg() { | 
					
						
							|  |  |  | 					encs = append(encs, d.Val()) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				if len(encs) == 0 { | 
					
						
							|  |  |  | 					return d.ArgErr() | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				enc.Prefer = encs | 
					
						
							|  |  |  | 			case "match": | 
					
						
							|  |  |  | 				err := enc.parseNamedResponseMatcher(d.NewFromNextSegment(), responseMatchers) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				matcher := responseMatchers["match"] | 
					
						
							|  |  |  | 				enc.Matcher = &matcher | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | 				name := d.Val() | 
					
						
							|  |  |  | 				modID := "http.encoders." + name | 
					
						
							|  |  |  | 				unm, err := caddyfile.UnmarshalModule(d, modID) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				encoding, ok := unm.(Encoding) | 
					
						
							|  |  |  | 				if !ok { | 
					
						
							|  |  |  | 					return d.Errf("module %s is not an HTTP encoding; is %T", modID, unm) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				if enc.EncodingsRaw == nil { | 
					
						
							|  |  |  | 					enc.EncodingsRaw = make(caddy.ModuleMap) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				enc.EncodingsRaw[name] = caddyconfig.JSON(encoding, nil) | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-30 02:47:19 +02:00
										 |  |  | // Parse the tokens of a named response matcher. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | //     match { | 
					
						
							|  |  |  | //         header <field> [<value>] | 
					
						
							|  |  |  | //         status <code...> | 
					
						
							|  |  |  | //     } | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Or, single line syntax: | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | //     match [header <field> [<value>]] | [status <code...>] | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | func (enc *Encode) parseNamedResponseMatcher(d *caddyfile.Dispenser, matchers map[string]caddyhttp.ResponseMatcher) error { | 
					
						
							|  |  |  | 	for d.Next() { | 
					
						
							|  |  |  | 		definitionName := d.Val() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if _, ok := matchers[definitionName]; ok { | 
					
						
							|  |  |  | 			return d.Errf("matcher is defined more than once: %s", definitionName) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		matcher := caddyhttp.ResponseMatcher{} | 
					
						
							|  |  |  | 		for nesting := d.Nesting(); d.NextArg() || d.NextBlock(nesting); { | 
					
						
							|  |  |  | 			switch d.Val() { | 
					
						
							|  |  |  | 			case "header": | 
					
						
							|  |  |  | 				if matcher.Headers == nil { | 
					
						
							|  |  |  | 					matcher.Headers = http.Header{} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// reuse the header request matcher's unmarshaler | 
					
						
							|  |  |  | 				headerMatcher := caddyhttp.MatchHeader(matcher.Headers) | 
					
						
							|  |  |  | 				err := headerMatcher.UnmarshalCaddyfile(d.NewFromNextSegment()) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				matcher.Headers = http.Header(headerMatcher) | 
					
						
							|  |  |  | 			case "status": | 
					
						
							|  |  |  | 				if matcher.StatusCode == nil { | 
					
						
							|  |  |  | 					matcher.StatusCode = []int{} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				args := d.RemainingArgs() | 
					
						
							|  |  |  | 				if len(args) == 0 { | 
					
						
							|  |  |  | 					return d.ArgErr() | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				for _, arg := range args { | 
					
						
							|  |  |  | 					if len(arg) == 3 && strings.HasSuffix(arg, "xx") { | 
					
						
							|  |  |  | 						arg = arg[:1] | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					statusNum, err := strconv.Atoi(arg) | 
					
						
							|  |  |  | 					if err != nil { | 
					
						
							|  |  |  | 						return d.Errf("bad status value '%s': %v", arg, err) | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					matcher.StatusCode = append(matcher.StatusCode, statusNum) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | 				return d.Errf("unrecognized response matcher %s", d.Val()) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		matchers[definitionName] = matcher | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | // Interface guard | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | var _ caddyfile.Unmarshaler = (*Encode)(nil) |