httpfilter is a package that allows you to bundle your website's content with the behavior that determines how the content is made visible by your web server.
On many websites the URL path that a user goes to is different from the internal path of the actual resource that the server responds with. For example, if you go to https://ryanburleson.dev/home
, the path visible to the user is /home
, but the path of the HTML file that is served is actually /home.html
.
Typically, the way to acheive this behavior is to hard code it into your web server. You may use a mapping function to define a request path and a corresponding resource path. This introduces two problems:
- The code that determines how a resource should be made visible is separated from the actual content.
- If you want to add new resources, you must modify your web server code.
httpfilter aims to solve both of these problems. httpfilter uses simple scripts that are embeded in the same file directory as your resources in order to bundle the content and behavior and allow you to dynamically change the behavior without modifying the server code.
This makes your web resources more modular, i.e. a directory containing some content also contains the code that specifies how that content should be displayed by a web server.
- Installation
- Tutorial
- How it Works
- Usage Example
- Standard Operators
- Writing Filters
- Attaching Additional Operators
- Writing Operator Functions
- Fixed Filter File
$ go get github.com/rbxb/httpfilter
$ go install github.com/rbxb/httpfilter/cmd/httpfilter
Playing around with the pre-made sample should help you get a feel for how httpfilter works.
httpfilter uses scripts (called filters) nested in the website's static files directories to invoke Go functions that modify how the request is handled. For example, if you wanted to make the request path /index.html
redirect to /home.html
, you could use a filter like this:
#redirect index.html home.html
This filter file would be placed in the same directory as the supposed index.html
file and the home.html
file.
When the httpfilter server recieves a request for /index.html
, it will first look for the filter in the same directory as the requested file. The server will read the filter from the top—down, searching for an entry where the selector matches the requested file name. If an entry's selector matches the requested file name, the server will call the Go function that is mapped to the operator of that entry. Any additional arguments are passed to the Go function as strings.
entry
┏━━━━━━━━━┻━━━━━━━━━━━━━━━━╍┅
#redirect index.html home.html
┃ ┃ ┗━━━┳━━╍┅
operator ┃ arguments
┃
selector
The httpfilter server passes the request and arguments to the operator function, which may write a response. The server will continue to read and execute entries from the filter until a response has been written.
If the server reaches the end of the filter and a response still hasn't been written, the server will call the serve
operator, which will attempt to serve the file or respond with a 404 Not found
error.
Create an httpfilter Server
and pass it to http.ListenAndServe
.
package main
import (
"log"
"net/http"
"github.com/rbxb/httpfilter"
)
func main() {
server := httpfilter.NewServer("./root", "")
log.Fatal(http.ListenAndServe(":8080", server))
}
The serve
operator attempts to serve the file named in the first argument or responds with a 404 Not found
error. E.g. this will serve the file home.html
when the client requests /home
:
#serve home home.html
If the request is not fulfilled at the end of the filter file, the httpfilter server will default to the serve operator to write a response.
The ignore
operator responds with a 404 Not Found
error. Use this if you want to prevent access to a specific file, e.g.
#ignore secret.txt
The redirect
operator redirects a request to the URL or path in the first argument. E.g. this will redirect /index.html
to /home
:
#redirect index.html home
The proxy
operator forwards the client's request to the end server provided in the first argument and responds with the end server's response:
#proxy page.html http://192.168.1.6:80
The request
makes a new GET request to the server in the first argument and responds with the end server's response:
#request page.html http://192.168.1.6:80
- Operators are always prefixed by a
#
. - Operators, selectors, and arguments are separated by spaces.
- Entries are separated by line breaks.
- The filter is read from the top—down and the server will never read upwards
You can use a *
in the selector to select all queries.
*
will match with all queries.a.*
will match with all queries where the name isa
regardless of the extension.*.a
will match with all queries where the extension is.a
.
For example, this filter will ignore
all queries where the extension is .txt
:
#ignore *.txt
Using the @
symbol selects a request by its subdomain.
For example, this filter will proxy the subdomain service
to a local server and redirect the base domain to the service
subdomain.
#proxy @service http://192.168.1.6:80
#redirect @ http://service.example.com
If you have multiple entries that use the same operator repeatedly, e.g.
#serve home home.html
#serve about about.html
#serve contact contact.html
you can write the operator once and put the entries below it:
#serve
home home.html
about about.html
contact contact.html
Currently, filter files must be named _filters.txt
.
Place the file in the directory that you want it to work in.
The httpfilter server will never serve a filter file to a client.
Non-standard operator functions can be attached to the server.
Pass them to NewServer
as a map[string]OpFunc
, where the map key is the operator name that should be used in the filter file to call the operator function.
func NewServer(root string, filter string, ops ...map[string]OpFunc) * Server
Pass in your own operator functions:
server := httpfilter.NewServer("./root", "", map[string]httpfilter.OpFunc{
"myop": myOpFunc,
})
The standard operators can be overwritten by passing in operators with the same key value.
You can write your own operator functions and attach them to your server (see Attaching Additional Operators).
Operator functions follow this type:
type OpFunc func(w http.ResponseWriter, req *http.Request, args ...string)
If the operator function calls w.Write
or w.WriteHeader
, the server will stop executing entries and the request/response is completed.
server := httpfilter.NewServer("./root", "C:/_filter.txt")
Putting a path into the second argument of the server constructor will force the server to use that filter file for every request. You can use a fixed filter file and the @
selector to route subdomains to other servers.