| 
									
										
										
										
											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 fileserver | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2019-12-12 15:27:09 -07:00
										 |  |  | 	"strings" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 13:36:46 -07:00
										 |  |  | 	"github.com/caddyserver/caddy/v2" | 
					
						
							| 
									
										
										
										
											2019-08-09 12:19:56 -06:00
										 |  |  | 	"github.com/caddyserver/caddy/v2/caddyconfig/httpcaddyfile" | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 	"github.com/caddyserver/caddy/v2/modules/caddyhttp" | 
					
						
							| 
									
										
										
										
											2019-08-27 14:38:24 -06:00
										 |  |  | 	"github.com/caddyserver/caddy/v2/modules/caddyhttp/rewrite" | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | func init() { | 
					
						
							|  |  |  | 	httpcaddyfile.RegisterHandlerDirective("file_server", parseCaddyfile) | 
					
						
							|  |  |  | 	httpcaddyfile.RegisterDirective("try_files", parseTryFiles) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-27 11:51:32 -07:00
										 |  |  | // parseCaddyfile parses the file_server directive. It enables the static file | 
					
						
							|  |  |  | // server and configures it with this syntax: | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | //    file_server [<matcher>] [browse] { | 
					
						
							|  |  |  | //        root   <path> | 
					
						
							|  |  |  | //	      hide   <files...> | 
					
						
							|  |  |  | //	      index  <files...> | 
					
						
							|  |  |  | //	      browse [<template_file>] | 
					
						
							|  |  |  | //    } | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | func parseCaddyfile(h httpcaddyfile.Helper) (caddyhttp.MiddlewareHandler, error) { | 
					
						
							|  |  |  | 	var fsrv FileServer | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for h.Next() { | 
					
						
							|  |  |  | 		args := h.RemainingArgs() | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 		switch len(args) { | 
					
						
							|  |  |  | 		case 0: | 
					
						
							|  |  |  | 		case 1: | 
					
						
							|  |  |  | 			if args[0] != "browse" { | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 				return nil, h.ArgErr() | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			fsrv.Browse = new(Browse) | 
					
						
							|  |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 			return nil, h.ArgErr() | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-10 19:21:52 -06:00
										 |  |  | 		for h.NextBlock(0) { | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 			switch h.Val() { | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 			case "hide": | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 				fsrv.Hide = h.RemainingArgs() | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 				if len(fsrv.Hide) == 0 { | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 					return nil, h.ArgErr() | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			case "index": | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 				fsrv.IndexNames = h.RemainingArgs() | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 				if len(fsrv.Hide) == 0 { | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 					return nil, h.ArgErr() | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			case "root": | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 				if !h.Args(&fsrv.Root) { | 
					
						
							|  |  |  | 					return nil, h.ArgErr() | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			case "browse": | 
					
						
							|  |  |  | 				if fsrv.Browse != nil { | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 					return nil, h.Err("browsing is already configured") | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				fsrv.Browse = new(Browse) | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 				h.Args(&fsrv.Browse.TemplateFile) | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 			default: | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 				return nil, h.Errf("unknown subdirective '%s'", h.Val()) | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-21 15:50:02 -06:00
										 |  |  | 	// hide the Caddyfile (and any imported Caddyfiles) | 
					
						
							|  |  |  | 	if configFiles := h.Caddyfiles(); len(configFiles) > 0 { | 
					
						
							|  |  |  | 		for _, file := range configFiles { | 
					
						
							|  |  |  | 			if !fileHidden(file, fsrv.Hide) { | 
					
						
							|  |  |  | 				fsrv.Hide = append(fsrv.Hide, file) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 	return &fsrv, nil | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-15 12:47:38 -07:00
										 |  |  | // parseTryFiles parses the try_files directive. It combines a file matcher | 
					
						
							|  |  |  | // with a rewrite directive, so this is not a standard handler directive. | 
					
						
							|  |  |  | // A try_files directive has this syntax (notice no matcher tokens accepted): | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | //    try_files <files...> | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2019-12-12 15:27:09 -07:00
										 |  |  | // and is basically shorthand for: | 
					
						
							| 
									
										
										
										
											2019-11-15 12:47:38 -07:00
										 |  |  | // | 
					
						
							|  |  |  | //    matcher:try_files { | 
					
						
							|  |  |  | //        file { | 
					
						
							|  |  |  | //            try_files <files...> | 
					
						
							|  |  |  | //        } | 
					
						
							|  |  |  | //    } | 
					
						
							|  |  |  | //    rewrite match:try_files {http.matchers.file.relative}{http.request.uri.query_string} | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2019-12-12 15:27:09 -07:00
										 |  |  | // If any of the files in the list have a query string, the query string will | 
					
						
							|  |  |  | // be ignored when checking for file existence, but will be augmented into | 
					
						
							|  |  |  | // the request's URI when rewriting the request. | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | func parseTryFiles(h httpcaddyfile.Helper) ([]httpcaddyfile.ConfigValue, error) { | 
					
						
							|  |  |  | 	if !h.Next() { | 
					
						
							|  |  |  | 		return nil, h.ArgErr() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-12 15:27:09 -07:00
										 |  |  | 	tryFiles := h.RemainingArgs() | 
					
						
							|  |  |  | 	if len(tryFiles) == 0 { | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 		return nil, h.ArgErr() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-12 15:27:09 -07:00
										 |  |  | 	// makeRoute returns a route that tries the files listed in try | 
					
						
							|  |  |  | 	// and then rewrites to the matched file, and appends writeURIAppend | 
					
						
							|  |  |  | 	// to the end of the query string. | 
					
						
							|  |  |  | 	makeRoute := func(try []string, writeURIAppend string) []httpcaddyfile.ConfigValue { | 
					
						
							|  |  |  | 		handler := rewrite.Rewrite{ | 
					
						
							| 
									
										
										
										
											2019-12-17 16:30:26 -07:00
										 |  |  | 			Rehandle: true, | 
					
						
							|  |  |  | 			URI:      "{http.matchers.file.relative}{http.request.uri.query_string}" + writeURIAppend, | 
					
						
							| 
									
										
										
										
											2019-12-12 15:27:09 -07:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		matcherSet := caddy.ModuleMap{ | 
					
						
							|  |  |  | 			"file": h.JSON(MatchFile{ | 
					
						
							|  |  |  | 				TryFiles: try, | 
					
						
							|  |  |  | 			}), | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return h.NewRoute(matcherSet, handler) | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-08-09 12:05:47 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-12 15:27:09 -07:00
										 |  |  | 	var result []httpcaddyfile.ConfigValue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// if there are query strings in the list, we have to split into | 
					
						
							|  |  |  | 	// a separate route for each item with a query string, because | 
					
						
							|  |  |  | 	// the rewrite is different for that item | 
					
						
							|  |  |  | 	var try []string | 
					
						
							|  |  |  | 	for _, item := range tryFiles { | 
					
						
							|  |  |  | 		if idx := strings.Index(item, "?"); idx >= 0 { | 
					
						
							|  |  |  | 			if len(try) > 0 { | 
					
						
							|  |  |  | 				result = append(result, makeRoute(try, "")...) | 
					
						
							|  |  |  | 				try = []string{} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			result = append(result, makeRoute([]string{item[:idx]}, "&"+item[idx+1:])...) | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// accumulate consecutive non-query-string parameters | 
					
						
							|  |  |  | 		try = append(try, item) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(try) > 0 { | 
					
						
							|  |  |  | 		result = append(result, makeRoute(try, "")...) | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-12 15:27:09 -07:00
										 |  |  | 	return result, nil | 
					
						
							| 
									
										
										
										
											2019-08-21 10:46:35 -06:00
										 |  |  | } |