Compare commits
6 Commits
Author | SHA1 | Date | |
---|---|---|---|
44cd195109 | |||
810cebadd9 | |||
761bd3a6f9 | |||
4ab5a7fd1d | |||
cc9e7d4f22 | |||
a1661a4122 |
133
README.md
133
README.md
@ -1,5 +1,5 @@
|
|||||||
# nginx-cache-purge
|
# nginx-cache-purge
|
||||||
A tool to help purge Nginx cache. It can either run locally with the purge command, or run as a local unix service to allow for purging by Nginx http requests.
|
A tool to help purge Nginx cache. It can either run locally with the purge command, or run as a local unix service to allow for purging by Nginx http requests. The tool supports using wildcard/glob syntax in the purge key to match multiple keys from the cache.
|
||||||
|
|
||||||
## Install
|
## Install
|
||||||
You can install either by downloading the latest binary release, or by building.
|
You can install either by downloading the latest binary release, or by building.
|
||||||
@ -10,6 +10,29 @@ Building should be as simple as running:
|
|||||||
go build
|
go build
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
The following are some examples of ways to purge cache
|
||||||
|
|
||||||
|
### Purge a specific key
|
||||||
|
```
|
||||||
|
$ nginx-cache-purge purge /var/nginx/proxy_temp/cache example.com/index.html
|
||||||
|
```
|
||||||
|
|
||||||
|
### Purge all keys for a domain
|
||||||
|
```
|
||||||
|
$ nginx-cache-purge purge /var/nginx/proxy_temp/cache 'example.com/*'
|
||||||
|
```
|
||||||
|
|
||||||
|
### Purge all keys for jpeg and png files
|
||||||
|
```
|
||||||
|
$ nginx-cache-purge purge /var/nginx/proxy_temp/cache 'example.com/*.{jpg,jpeg,png}'
|
||||||
|
```
|
||||||
|
|
||||||
|
### Purge all keys
|
||||||
|
```
|
||||||
|
$ nginx-cache-purge purge /var/nginx/proxy_temp/cache '*'
|
||||||
|
```
|
||||||
|
|
||||||
## Running as a service
|
## Running as a service
|
||||||
If you want to run as a service to allow purge requests via http requests, you'll need to create a systemd service file and place it in `/etc/systemd/system/nginx-cache-purge.service`.
|
If you want to run as a service to allow purge requests via http requests, you'll need to create a systemd service file and place it in `/etc/systemd/system/nginx-cache-purge.service`.
|
||||||
```
|
```
|
||||||
@ -21,7 +44,6 @@ After=network.target
|
|||||||
User=nginx
|
User=nginx
|
||||||
Group=nginx
|
Group=nginx
|
||||||
RuntimeDirectory=nginx-cache-purge
|
RuntimeDirectory=nginx-cache-purge
|
||||||
PIDFile=/var/run/nginx-cache-purge/service.pid
|
|
||||||
ExecStart=/usr/local/bin/nginx-cache-purge server
|
ExecStart=/usr/local/bin/nginx-cache-purge server
|
||||||
Restart=always
|
Restart=always
|
||||||
RestartSec=3s
|
RestartSec=3s
|
||||||
@ -48,13 +70,14 @@ http {
|
|||||||
}
|
}
|
||||||
|
|
||||||
proxy_cache_path /var/nginx/proxy_temp/cache levels=1:2 keys_zone=my_cache:10m;
|
proxy_cache_path /var/nginx/proxy_temp/cache levels=1:2 keys_zone=my_cache:10m;
|
||||||
proxy_cache_key $host$request_uri;
|
proxy_cache_key $server_name$request_uri;
|
||||||
|
|
||||||
server {
|
server {
|
||||||
location / {
|
location / {
|
||||||
|
proxy_cache_bypass $is_purge;
|
||||||
if ($is_purge) {
|
if ($is_purge) {
|
||||||
proxy_pass http://unix:/var/run/nginx-cache-purge/http.sock;
|
proxy_pass http://unix:/run/nginx-cache-purge/http.sock;
|
||||||
rewrite ^ /?path=/var/nginx/proxy_temp/cache&key=$host$request_uri break;
|
rewrite ^ /?path=/var/nginx/proxy_temp/cache&key=$server_name$request_uri break;
|
||||||
}
|
}
|
||||||
|
|
||||||
proxy_cache my_cache;
|
proxy_cache my_cache;
|
||||||
@ -73,13 +96,14 @@ http {
|
|||||||
}
|
}
|
||||||
|
|
||||||
proxy_cache_path /var/nginx/proxy_temp/cache levels=1:2 keys_zone=my_cache:10m;
|
proxy_cache_path /var/nginx/proxy_temp/cache levels=1:2 keys_zone=my_cache:10m;
|
||||||
proxy_cache_key $host$request_uri;
|
proxy_cache_key $server_name$request_uri;
|
||||||
|
|
||||||
server {
|
server {
|
||||||
location / {
|
location / {
|
||||||
|
proxy_cache_bypass $is_purge;
|
||||||
if ($is_purge) {
|
if ($is_purge) {
|
||||||
proxy_pass http://unix:/var/run/nginx-cache-purge/http.sock;
|
proxy_pass http://unix:/run/nginx-cache-purge/http.sock;
|
||||||
rewrite ^ /?path=/var/nginx/proxy_temp/cache&key=$host$request_uri break;
|
rewrite ^ /?path=/var/nginx/proxy_temp/cache&key=$server_name$request_uri break;
|
||||||
}
|
}
|
||||||
|
|
||||||
proxy_cache my_cache;
|
proxy_cache my_cache;
|
||||||
@ -98,13 +122,50 @@ http {
|
|||||||
}
|
}
|
||||||
|
|
||||||
proxy_cache_path /var/nginx/proxy_temp/cache levels=1:2 keys_zone=my_cache:10m;
|
proxy_cache_path /var/nginx/proxy_temp/cache levels=1:2 keys_zone=my_cache:10m;
|
||||||
proxy_cache_key $host$request_uri;
|
proxy_cache_key $server_name$request_uri;
|
||||||
|
|
||||||
server {
|
server {
|
||||||
location / {
|
location / {
|
||||||
|
proxy_cache_bypass $is_purge;
|
||||||
if ($is_purge) {
|
if ($is_purge) {
|
||||||
proxy_pass http://unix:/var/run/nginx-cache-purge/http.sock;
|
proxy_pass http://unix:/run/nginx-cache-purge/http.sock;
|
||||||
rewrite ^ /?path=/var/nginx/proxy_temp/cache&key=$host$request_uri break;
|
rewrite ^ /?path=/var/nginx/proxy_temp/cache&key=$server_name$request_uri break;
|
||||||
|
}
|
||||||
|
|
||||||
|
proxy_cache my_cache;
|
||||||
|
proxy_pass http://upstream;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Auth via header and IP white list.
|
||||||
|
```
|
||||||
|
http {
|
||||||
|
map $http_purge_token $is_purge {
|
||||||
|
default 0;
|
||||||
|
nnCgKUx1p2bIABXR 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
geo $purge_allowed {
|
||||||
|
default 0;
|
||||||
|
127.0.0.1 1;
|
||||||
|
192.168.0.0/24 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
proxy_cache_path /var/nginx/proxy_temp/cache levels=1:2 keys_zone=my_cache:10m;
|
||||||
|
proxy_cache_key $server_name$request_uri;
|
||||||
|
|
||||||
|
server {
|
||||||
|
location / {
|
||||||
|
set $should_purge $purge_allowed;
|
||||||
|
if ($is_purge != 1) {
|
||||||
|
set $should_purge 0;
|
||||||
|
}
|
||||||
|
proxy_cache_bypass $should_purge;
|
||||||
|
if ($should_purge) {
|
||||||
|
proxy_pass http://unix:/run/nginx-cache-purge/http.sock;
|
||||||
|
rewrite ^ /?path=/var/nginx/proxy_temp/cache&key=$server_name$request_uri break;
|
||||||
}
|
}
|
||||||
|
|
||||||
proxy_cache my_cache;
|
proxy_cache my_cache;
|
||||||
@ -118,7 +179,7 @@ http {
|
|||||||
```
|
```
|
||||||
http {
|
http {
|
||||||
proxy_cache_path /var/nginx/proxy_temp/cache levels=1:2 keys_zone=my_cache:10m;
|
proxy_cache_path /var/nginx/proxy_temp/cache levels=1:2 keys_zone=my_cache:10m;
|
||||||
proxy_cache_key $host$request_uri;
|
proxy_cache_key $server_name$request_uri;
|
||||||
|
|
||||||
server {
|
server {
|
||||||
location / {
|
location / {
|
||||||
@ -128,9 +189,53 @@ http {
|
|||||||
location ~ /purge(/.*) {
|
location ~ /purge(/.*) {
|
||||||
allow 127.0.0.1;
|
allow 127.0.0.1;
|
||||||
deny all;
|
deny all;
|
||||||
proxy_pass http://unix:/var/run/nginx-cache-purge/http.sock;
|
proxy_pass http://unix:/run/nginx-cache-purge/http.sock;
|
||||||
rewrite ^ /?path=/var/nginx/proxy_temp/cache&key=$host$1 break;
|
rewrite ^ /?path=/var/nginx/proxy_temp/cache&key=$server_name$1 break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Help
|
||||||
|
```
|
||||||
|
$ nginx-cache-purge --help
|
||||||
|
Usage: nginx-cache-purge <command> [flags]
|
||||||
|
|
||||||
|
Tool to help purge cache from Nginx
|
||||||
|
|
||||||
|
Flags:
|
||||||
|
-h, --help Show context-sensitive help.
|
||||||
|
--version Print version information and quit
|
||||||
|
|
||||||
|
Commands:
|
||||||
|
server (s) Run the server
|
||||||
|
purge (p) Purge cache now
|
||||||
|
|
||||||
|
Run "nginx-cache-purge <command> --help" for more information on a command.
|
||||||
|
|
||||||
|
$ nginx-cache-purge p --help
|
||||||
|
Usage: nginx-cache-purge purge (p) <cache-path> <key> [flags]
|
||||||
|
|
||||||
|
Purge cache now
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
<cache-path> Path to cache directory.
|
||||||
|
<key> Cache key or wildcard match.
|
||||||
|
|
||||||
|
Flags:
|
||||||
|
-h, --help Show context-sensitive help.
|
||||||
|
--version Print version information and quit
|
||||||
|
|
||||||
|
--exclude-key=EXCLUDE-KEY,... Key to exclude, can be wild card and can add multiple excludes.
|
||||||
|
|
||||||
|
$ nginx-cache-purge s --help
|
||||||
|
Usage: nginx-cache-purge server (s) [flags]
|
||||||
|
|
||||||
|
Run the server
|
||||||
|
|
||||||
|
Flags:
|
||||||
|
-h, --help Show context-sensitive help.
|
||||||
|
--version Print version information and quit
|
||||||
|
|
||||||
|
--socket=STRING Socket path for HTTP communication.
|
||||||
|
``
|
||||||
|
24
main.go
24
main.go
@ -5,6 +5,7 @@ import (
|
|||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
@ -17,7 +18,7 @@ import (
|
|||||||
const (
|
const (
|
||||||
serviceName = "nginx-cache-purge"
|
serviceName = "nginx-cache-purge"
|
||||||
serviceDescription = "Tool to help purge Nginx cache "
|
serviceDescription = "Tool to help purge Nginx cache "
|
||||||
serviceVersion = "0.1"
|
serviceVersion = "0.1.4"
|
||||||
)
|
)
|
||||||
|
|
||||||
// App structure to access global app variables.
|
// App structure to access global app variables.
|
||||||
@ -42,7 +43,7 @@ func (a *App) PurgeCache(CachePath string, Key string, ExcludeKeys []string) err
|
|||||||
for _, exclude := range ExcludeKeys {
|
for _, exclude := range ExcludeKeys {
|
||||||
if globRegex.MatchString(exclude) {
|
if globRegex.MatchString(exclude) {
|
||||||
g, err := glob.Compile(exclude)
|
g, err := glob.Compile(exclude)
|
||||||
if err != nil && g != nil && g.Match(Key) {
|
if err == nil && g != nil && g.Match(Key) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -62,7 +63,7 @@ func (a *App) PurgeCache(CachePath string, Key string, ExcludeKeys []string) err
|
|||||||
if !globRegex.MatchString(Key) {
|
if !globRegex.MatchString(Key) {
|
||||||
// If excluded, skip the key.
|
// If excluded, skip the key.
|
||||||
if keyIsExcluded(Key) {
|
if keyIsExcluded(Key) {
|
||||||
fmt.Println("Key", Key, "is excluded, will not purge.")
|
log.Println("Key", Key, "is excluded, will not purge.")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -82,7 +83,7 @@ func (a *App) PurgeCache(CachePath string, Key string, ExcludeKeys []string) err
|
|||||||
}
|
}
|
||||||
// If this file matches our key hash then delete.
|
// If this file matches our key hash then delete.
|
||||||
if info.Name() == keyHash {
|
if info.Name() == keyHash {
|
||||||
fmt.Printf("Purging %s as it matches the key %s requested to be purged.\n", filePath, Key)
|
log.Printf("Purging %s as it matches the key %s requested to be purged.\n", filePath, Key)
|
||||||
err := os.Remove(filePath)
|
err := os.Remove(filePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
@ -118,12 +119,21 @@ func (a *App) PurgeCache(CachePath string, Key string, ExcludeKeys []string) err
|
|||||||
}
|
}
|
||||||
defer file.Close()
|
defer file.Close()
|
||||||
scanner := bufio.NewScanner(file)
|
scanner := bufio.NewScanner(file)
|
||||||
|
// Scan file for the key.
|
||||||
for scanner.Scan() {
|
for scanner.Scan() {
|
||||||
line := scanner.Text()
|
line := scanner.Text()
|
||||||
|
// If line is the key, check if it matches our glob pattern and delete.
|
||||||
if strings.HasPrefix(line, "KEY: ") {
|
if strings.HasPrefix(line, "KEY: ") {
|
||||||
key := line[5:]
|
keyRead := line[5:]
|
||||||
if g.Match(key) {
|
if g.Match(keyRead) {
|
||||||
fmt.Printf("Purging %s as it matches the key %s requested to be purged.\n", filePath, Key)
|
// If excluded, skip the key.
|
||||||
|
if keyIsExcluded(keyRead) {
|
||||||
|
log.Println("Key", keyRead, "is excluded, will not purge.")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete the file.
|
||||||
|
log.Printf("Purging %s as it matches the key %s requested to be purged.\n", filePath, Key)
|
||||||
err := os.Remove(filePath)
|
err := os.Remove(filePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -3,6 +3,7 @@ package main
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"log"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
@ -50,7 +51,7 @@ func (a *ServerCmd) Run() error {
|
|||||||
// Determine UNIX socket path.
|
// Determine UNIX socket path.
|
||||||
unixSocket := a.Socket
|
unixSocket := a.Socket
|
||||||
if unixSocket == "" {
|
if unixSocket == "" {
|
||||||
unixSocket = "/var/run/nginx-cache-purge/http.sock"
|
unixSocket = "/run/nginx-cache-purge/http.sock"
|
||||||
}
|
}
|
||||||
|
|
||||||
// If socket exists, remove it.
|
// If socket exists, remove it.
|
||||||
@ -66,7 +67,7 @@ func (a *ServerCmd) Run() error {
|
|||||||
defer listener.Close()
|
defer listener.Close()
|
||||||
|
|
||||||
// Start the FastCGI server.
|
// Start the FastCGI server.
|
||||||
fmt.Println("Starting server at", unixSocket)
|
log.Println("Starting server at", unixSocket)
|
||||||
http.HandleFunc("/", a.ServeHTTP)
|
http.HandleFunc("/", a.ServeHTTP)
|
||||||
err = http.Serve(listener, nil)
|
err = http.Serve(listener, nil)
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user