├── .clj-kondo └── config.edn ├── .dockerignore ├── dev.env ├── resources └── simplelogger.properties ├── Dockerfile ├── docker-push.sh ├── .gitignore ├── deps.edn ├── test └── ivarref │ └── http_test.clj ├── LICENSE ├── README.md └── src └── com └── github └── ivarref └── mikkmokk_proxy.clj /.clj-kondo/config.edn: -------------------------------------------------------------------------------- 1 | {:lint-as {dom-top.core/letr clojure.core/let}} 2 | -------------------------------------------------------------------------------- /.dockerignore: -------------------------------------------------------------------------------- 1 | * 2 | !src/ 3 | !.git/ 4 | !resources/ 5 | !deps.edn 6 | !build.edn 7 | -------------------------------------------------------------------------------- /dev.env: -------------------------------------------------------------------------------- 1 | MIKKMOKK_DEVELOPMENT=true 2 | #DESTINATION_URL=http://example.com 3 | 4 | ADMIN_BIND=0.0.0.0 5 | ADMIN_PORT=7070 6 | PROXY_BIND=0.0.0.0 7 | PROXY_PORT=8080 8 | 9 | TZ=Europe/Oslo 10 | -------------------------------------------------------------------------------- /resources/simplelogger.properties: -------------------------------------------------------------------------------- 1 | org.slf4j.simpleLogger.defaultLogLevel=info 2 | #org.slf4j.simpleLogger.log.xxxxx= 3 | org.slf4j.simpleLogger.showDateTime=true 4 | #org.slf4j.simpleLogger.dateTimeFormat=HH:mm:ss 5 | org.slf4j.simpleLogger.dateTimeFormat=yyyy-MM-dd HH:mm:ss:SSS Z 6 | org.slf4j.simpleLogger.showThreadName=false 7 | org.slf4j.simpleLogger.levelInBrackets=true 8 | org.slf4j.simpleLogger.showLogName=false 9 | org.slf4j.simpleLogger.showShortLogName=false 10 | -------------------------------------------------------------------------------- /Dockerfile: -------------------------------------------------------------------------------- 1 | FROM clojure:temurin-17-tools-deps-1.11.1.1113-focal as builder 2 | WORKDIR /src 3 | COPY deps.edn . 4 | COPY build.edn . 5 | RUN clojure -P && clojure -P -T:build 6 | COPY src/ src/ 7 | COPY .git/ .git/ 8 | COPY resources/ resources/ 9 | RUN clojure -T:build uberjar 10 | 11 | FROM eclipse-temurin:22.0.1_8-jdk-jammy 12 | COPY --from=builder /src/target/mikkmokk-proxy-standalone.jar /mikkmokk-proxy-standalone.jar 13 | ENTRYPOINT ["java", "-jar", "/mikkmokk-proxy-standalone.jar"] 14 | -------------------------------------------------------------------------------- /docker-push.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | set -ex 4 | 5 | git update-index --refresh 6 | git diff-index --quiet HEAD -- 7 | 8 | VERSION="v0.1.$(git rev-list --count HEAD)" 9 | 10 | echo "Releasing $VERSION" 11 | 12 | docker buildx create --name multiarch --driver docker-container --use || true 13 | docker buildx build --push --platform linux/arm64,linux/amd64 --tag docker.io/ivarref/mikkmokk-proxy:"$VERSION" . 14 | 15 | git tag -a "$VERSION" -m "Release $VERSION" 16 | git push --follow-tags 17 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .calva/output-window/ 2 | .classpath 3 | .clj-kondo/.cache 4 | .clj-kondo/clj-commons 5 | .cpcache 6 | .DS_Store 7 | .eastwood 8 | .factorypath 9 | .hg/ 10 | .hgignore 11 | .java-version 12 | .lein-* 13 | .lsp/.cache 14 | .lsp/sqlite.db 15 | .nrepl-history 16 | .nrepl-port 17 | .portal 18 | .project 19 | .rebel_readline_history 20 | .settings 21 | .socket-repl-port 22 | .sw* 23 | .vscode 24 | *.class 25 | *.jar 26 | *.swp 27 | *~ 28 | /checkouts 29 | /classes 30 | /target 31 | .idea/ 32 | *.iml 33 | -------------------------------------------------------------------------------- /deps.edn: -------------------------------------------------------------------------------- 1 | {:paths ["src" "resources"] 2 | :deps {org.clojure/clojure {:mvn/version "1.11.1"} 3 | aleph/aleph {:mvn/version "0.6.1"} 4 | org.slf4j/slf4j-simple {:mvn/version "1.7.36"} 5 | lambdaisland/regal {:mvn/version "0.0.143"} 6 | dom-top/dom-top {:mvn/version "1.0.8"} 7 | clojure-term-colors/clojure-term-colors {:mvn/version "0.1.0"}} 8 | :aliases {:build {:deps {com.github.liquidz/build.edn {:git/tag "0.3.90" :git/sha "e3a3e31"}} 9 | :ns-default build-edn.main} 10 | :test {:extra-paths ["test"] 11 | :extra-deps {io.github.cognitect-labs/test-runner 12 | {:git/tag "v0.5.1" :git/sha "dfb30dd"}} 13 | :main-opts ["-m" "cognitect.test-runner"] 14 | :exec-fn cognitect.test-runner.api/test}}} 15 | -------------------------------------------------------------------------------- /test/ivarref/http_test.clj: -------------------------------------------------------------------------------- 1 | (ns ivarref.http-test 2 | (:require [aleph.http :as http] 3 | [aleph.netty :as netty] 4 | [clj-commons.byte-streams :as bs] 5 | [clojure.test :refer [deftest is use-fixtures]] 6 | [clojure.tools.logging :as log] 7 | [com.github.ivarref.mikkmokk-proxy :as mm]) 8 | (:import (java.util.concurrent Executors))) 9 | 10 | (def ^:dynamic *server-port* nil) 11 | (def ^:dynamic *admin-port* nil) 12 | 13 | (defn with-server [f] 14 | (let [state {:one-off (atom #{}) 15 | :env {}} 16 | server (http/start-server (fn [req] (mm/outer-handler state req)) {:executor (Executors/newFixedThreadPool 8) :port 0}) 17 | admin (http/start-server (fn [req] (mm/admin-handler state req)) {:executor (Executors/newFixedThreadPool 2) :port 0})] 18 | (try 19 | (with-redefs [mm/single-request (fn [_request-method url headers _body] 20 | {:status 200 21 | :body url 22 | :headers (dissoc headers "content-length")})] 23 | (binding [*server-port* (netty/port server) 24 | *admin-port* (netty/port admin)] 25 | (f))) 26 | (finally 27 | (try 28 | (.close server) 29 | (catch Throwable t 30 | (log/error t "Error while trying to close server:" (ex-message t)))) 31 | (try 32 | (.close admin) 33 | (catch Throwable t 34 | (log/error t "Error while trying to close admin server:" (ex-message t)))))))) 35 | 36 | (use-fixtures :each with-server) 37 | 38 | (defn get-uri [uri headers] 39 | (try 40 | (-> @(http/get (str "http://localhost:" *server-port* uri) {:headers headers}) 41 | (update :body bs/to-string)) 42 | (catch Throwable t 43 | (-> (ex-data t) 44 | (update :body bs/to-string))))) 45 | 46 | (defn post-admin-uri [uri headers] 47 | (try 48 | (-> @(http/post (str "http://localhost:" *admin-port* uri) {:headers headers}) 49 | (update :body bs/to-string)) 50 | (catch Throwable t 51 | (ex-data t)))) 52 | 53 | (deftest basic 54 | (is (= 500 (:status (get-uri "/" {})))) 55 | (is (= "http://example.com/" (:body (get-uri "/" {"x-mikkmokk-destination-url" "http://example.com"})))) 56 | (is (= 200 (:status (get-uri "/" {"x-mikkmokk-destination-url" "http://example.com"})))) 57 | (is (= "http://example.com/" (:body (get-uri "/mikkmokk-fwd-http/example.com" {})))) 58 | (is (= "http://example.com/" (:body (get-uri "/mikkmokk-fwd-http/example.com/" {})))) 59 | (is (= "http://example.com/" (:body (get-uri "/mikkmokk-forward-http/example.com/" {}))))) 60 | 61 | (deftest match-uri-starts-with 62 | (is (= 200 (:status (get-uri "/no-match" {"x-mikkmokk-destination-url" "http://example.com" 63 | "x-mikkmokk-match-uri-starts-with" "/match" 64 | "x-mikkmokk-fail-before-percentage" "100"})))) 65 | (is (= 503 (:status (get-uri "/match" {"x-mikkmokk-destination-url" "http://example.com" 66 | "x-mikkmokk-match-uri-starts-with" "/match" 67 | "x-mikkmokk-fail-before-percentage" "100"})))) 68 | (is (= 503 (:status (get-uri "/match/more" {"x-mikkmokk-destination-url" "http://example.com" 69 | "x-mikkmokk-match-uri-starts-with" "/match" 70 | "x-mikkmokk-fail-before-percentage" "100"}))))) 71 | 72 | (deftest match-header 73 | (is (= 200 (:status (get-uri "/" {"x-mikkmokk-destination-url" "http://example.com" 74 | "x-mikkmokk-match-header-name" "x-user-id" 75 | "x-mikkmokk-match-header-value" "some-user-id" 76 | "x-mikkmokk-fail-before-percentage" "100"})))) 77 | (is (= 503 (:status (get-uri "/" {"x-mikkmokk-destination-url" "http://example.com" 78 | "x-mikkmokk-match-header-name" "x-user-id" 79 | "x-mikkmokk-match-header-value" "some-user-id" 80 | "x-user-id" "some-user-id" 81 | "x-mikkmokk-fail-before-percentage" "100"}))))) 82 | 83 | 84 | (deftest match-host 85 | (is (= 200 (:status (get-uri "/mikkmokk-forward-http/example.com/" 86 | {"x-mikkmokk-match-host" "peggy.gmbh.com" 87 | "x-mikkmokk-fail-before-percentage" "100"})))) 88 | (is (= 503 (:status (get-uri "/mikkmokk-forward-http/example.com/" 89 | {"x-mikkmokk-match-host" "example.com" 90 | "x-mikkmokk-fail-before-percentage" "100"})))) 91 | (is (= 200 (:status (get-uri "/mikkmokk-forward-http/peggy.gmbh.com/some-endpoint" 92 | {"x-mikkmokk-match-host" "peggy.gmbh.com" 93 | "x-mikkmokk-match-uri" "/some-endpoint2" 94 | "x-mikkmokk-fail-before-percentage" "100"})))) 95 | (is (= 503 (:status (get-uri "/mikkmokk-forward-http/peggy.gmbh.com/some-endpoint" 96 | {"x-mikkmokk-match-host" "peggy.gmbh.com" 97 | "x-mikkmokk-match-uri" "/some-endpoint" 98 | "x-mikkmokk-fail-before-percentage" "100"}))))) 99 | 100 | 101 | 102 | (deftest one-off 103 | (is (= "http://example.com/" (:body (get-uri "/mikkmokk-forward-http/example.com/" {})))) 104 | (is (= 200 (:status (post-admin-uri "/api/v1/one-off" {"x-mikkmokk-fail-before-percentage" "100"})))) 105 | (is (= 503 (:status (get-uri "/mikkmokk-forward-http/example.com/" {})))) 106 | (is (= 200 (:status (get-uri "/mikkmokk-forward-http/example.com/" {})))) 107 | 108 | (is (= 200 (:status (post-admin-uri "/api/v1/one-off" {"x-mikkmokk-fail-before-percentage" "100"})))) 109 | (is (= 200 (:status (post-admin-uri "/api/v1/one-off" {"x-mikkmokk-fail-before-percentage" "100"})))) 110 | (is (= 503 (:status (get-uri "/mikkmokk-forward-http/example.com/" {})))) 111 | (is (= 503 (:status (get-uri "/mikkmokk-forward-http/example.com/" {})))) 112 | (is (= 200 (:status (get-uri "/mikkmokk-forward-http/example.com/" {})))) 113 | 114 | (is (= 200 (:status (post-admin-uri "/api/v1/one-off" {"x-mikkmokk-match-host" "example.com" 115 | "x-mikkmokk-match-uri" "/some-endpoint" 116 | "x-mikkmokk-fail-before-percentage" "100"})))) 117 | (is (= 200 (:status (get-uri "/mikkmokk-forward-http/example.com/some-endpoint2" {})))) 118 | (is (= 200 (:status (get-uri "/mikkmokk-forward-http/peggy.gmbh.com/some-endpoint" {})))) 119 | (is (= 503 (:status (get-uri "/mikkmokk-forward-http/example.com/some-endpoint" {})))) 120 | #_(is (= 503 (:status (get-uri "/" {"x-mikkmokk-destination-url" "http://example.com" 121 | "x-mikkmokk-match-header-name" "x-user-id" 122 | "x-mikkmokk-match-header-value" "some-user-id" 123 | "x-user-id" "some-user-id" 124 | "x-mikkmokk-fail-before-percentage" "100"}))))) 125 | 126 | (defn origin [resp] 127 | (get-in resp [:headers "origin"])) 128 | 129 | (deftest modified-headers 130 | (is (= {"host" "example.com"} 131 | (-> (get-uri "/mikkmokk-forward-http/example.com/" {}) 132 | :headers 133 | (select-keys ["host" "origin"])))) 134 | (is (= {"host" "example.com" 135 | "origin" "http://example.com"} 136 | (-> (get-uri "/mikkmokk-forward-http/example.com/" {"origin" "http://localhost:8090"}) 137 | :headers 138 | (select-keys ["host" "origin"])))) 139 | (is (= "http://example.com:8080" 140 | (origin (get-uri "/mikkmokk-forward-http/example.com:8080/" {"origin" "http://localhost:8090"})))) 141 | (is (= "https://example.com:8080" 142 | (origin (get-uri "/mikkmokk-forward-https/example.com:8080/" {"origin" "http://localhost:8090"})))) 143 | (is (= "https://example.com:8080" 144 | (origin (get-uri "/mikkmokk-forward-https/example.com:8080/api" {"origin" "http://localhost:8090"}))))) 145 | 146 | (deftest test-Access-Control-Allow-Origin 147 | (is (= "http://localhost:8090" 148 | (-> (get-in (get-uri "/mikkmokk-forward-http/example.com:8080/" {"origin" "http://localhost:8090" 149 | "Access-Control-Allow-Origin" "demo"}) 150 | [:headers "Access-Control-ALLOW-Origin"]))))) 151 | 152 | 153 | (deftest matches-uri-regex 154 | (is (= 200 (:status (get-uri "/mikkmokk-forward-http/example.com/a123123123" 155 | {"x-mikkmokk-match-uri-regex" "/[0-9]+" 156 | "x-mikkmokk-fail-before-percentage" "100"})))) 157 | (is (= 503 (:status (get-uri "/mikkmokk-forward-http/example.com/123123123" 158 | {"x-mikkmokk-match-uri-regex" "/[0-9]+" 159 | "x-mikkmokk-fail-before-percentage" "100"})))) 160 | (is (= 503 (:status (get-uri "/mikkmokk-forward-http/example.com/api/uuid/af9facf3-f679-4245-aa83-1b95cea52a1d" 161 | {"x-mikkmokk-match-uri-regex" "/api/uuid/([a-f0-9]{8}(-[a-f0-9]{4}){3}-[a-f0-9]{12})" 162 | "x-mikkmokk-fail-before-percentage" "100"}))))) 163 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Eclipse Public License - v 2.0 2 | 3 | THE ACCOMPANYING PROGRAM IS PROVIDED UNDER THE TERMS OF THIS ECLIPSE 4 | PUBLIC LICENSE ("AGREEMENT"). ANY USE, REPRODUCTION OR DISTRIBUTION 5 | OF THE PROGRAM CONSTITUTES RECIPIENT'S ACCEPTANCE OF THIS AGREEMENT. 6 | 7 | 1. DEFINITIONS 8 | 9 | "Contribution" means: 10 | 11 | a) in the case of the initial Contributor, the initial content 12 | Distributed under this Agreement, and 13 | 14 | b) in the case of each subsequent Contributor: 15 | i) changes to the Program, and 16 | ii) additions to the Program; 17 | where such changes and/or additions to the Program originate from 18 | and are Distributed by that particular Contributor. A Contribution 19 | "originates" from a Contributor if it was added to the Program by 20 | such Contributor itself or anyone acting on such Contributor's behalf. 21 | Contributions do not include changes or additions to the Program that 22 | are not Modified Works. 23 | 24 | "Contributor" means any person or entity that Distributes the Program. 25 | 26 | "Licensed Patents" mean patent claims licensable by a Contributor which 27 | are necessarily infringed by the use or sale of its Contribution alone 28 | or when combined with the Program. 29 | 30 | "Program" means the Contributions Distributed in accordance with this 31 | Agreement. 32 | 33 | "Recipient" means anyone who receives the Program under this Agreement 34 | or any Secondary License (as applicable), including Contributors. 35 | 36 | "Derivative Works" shall mean any work, whether in Source Code or other 37 | form, that is based on (or derived from) the Program and for which the 38 | editorial revisions, annotations, elaborations, or other modifications 39 | represent, as a whole, an original work of authorship. 40 | 41 | "Modified Works" shall mean any work in Source Code or other form that 42 | results from an addition to, deletion from, or modification of the 43 | contents of the Program, including, for purposes of clarity any new file 44 | in Source Code form that contains any contents of the Program. Modified 45 | Works shall not include works that contain only declarations, 46 | interfaces, types, classes, structures, or files of the Program solely 47 | in each case in order to link to, bind by name, or subclass the Program 48 | or Modified Works thereof. 49 | 50 | "Distribute" means the acts of a) distributing or b) making available 51 | in any manner that enables the transfer of a copy. 52 | 53 | "Source Code" means the form of a Program preferred for making 54 | modifications, including but not limited to software source code, 55 | documentation source, and configuration files. 56 | 57 | "Secondary License" means either the GNU General Public License, 58 | Version 2.0, or any later versions of that license, including any 59 | exceptions or additional permissions as identified by the initial 60 | Contributor. 61 | 62 | 2. GRANT OF RIGHTS 63 | 64 | a) Subject to the terms of this Agreement, each Contributor hereby 65 | grants Recipient a non-exclusive, worldwide, royalty-free copyright 66 | license to reproduce, prepare Derivative Works of, publicly display, 67 | publicly perform, Distribute and sublicense the Contribution of such 68 | Contributor, if any, and such Derivative Works. 69 | 70 | b) Subject to the terms of this Agreement, each Contributor hereby 71 | grants Recipient a non-exclusive, worldwide, royalty-free patent 72 | license under Licensed Patents to make, use, sell, offer to sell, 73 | import and otherwise transfer the Contribution of such Contributor, 74 | if any, in Source Code or other form. This patent license shall 75 | apply to the combination of the Contribution and the Program if, at 76 | the time the Contribution is added by the Contributor, such addition 77 | of the Contribution causes such combination to be covered by the 78 | Licensed Patents. The patent license shall not apply to any other 79 | combinations which include the Contribution. No hardware per se is 80 | licensed hereunder. 81 | 82 | c) Recipient understands that although each Contributor grants the 83 | licenses to its Contributions set forth herein, no assurances are 84 | provided by any Contributor that the Program does not infringe the 85 | patent or other intellectual property rights of any other entity. 86 | Each Contributor disclaims any liability to Recipient for claims 87 | brought by any other entity based on infringement of intellectual 88 | property rights or otherwise. As a condition to exercising the 89 | rights and licenses granted hereunder, each Recipient hereby 90 | assumes sole responsibility to secure any other intellectual 91 | property rights needed, if any. For example, if a third party 92 | patent license is required to allow Recipient to Distribute the 93 | Program, it is Recipient's responsibility to acquire that license 94 | before distributing the Program. 95 | 96 | d) Each Contributor represents that to its knowledge it has 97 | sufficient copyright rights in its Contribution, if any, to grant 98 | the copyright license set forth in this Agreement. 99 | 100 | e) Notwithstanding the terms of any Secondary License, no 101 | Contributor makes additional grants to any Recipient (other than 102 | those set forth in this Agreement) as a result of such Recipient's 103 | receipt of the Program under the terms of a Secondary License 104 | (if permitted under the terms of Section 3). 105 | 106 | 3. REQUIREMENTS 107 | 108 | 3.1 If a Contributor Distributes the Program in any form, then: 109 | 110 | a) the Program must also be made available as Source Code, in 111 | accordance with section 3.2, and the Contributor must accompany 112 | the Program with a statement that the Source Code for the Program 113 | is available under this Agreement, and informs Recipients how to 114 | obtain it in a reasonable manner on or through a medium customarily 115 | used for software exchange; and 116 | 117 | b) the Contributor may Distribute the Program under a license 118 | different than this Agreement, provided that such license: 119 | i) effectively disclaims on behalf of all other Contributors all 120 | warranties and conditions, express and implied, including 121 | warranties or conditions of title and non-infringement, and 122 | implied warranties or conditions of merchantability and fitness 123 | for a particular purpose; 124 | 125 | ii) effectively excludes on behalf of all other Contributors all 126 | liability for damages, including direct, indirect, special, 127 | incidental and consequential damages, such as lost profits; 128 | 129 | iii) does not attempt to limit or alter the recipients' rights 130 | in the Source Code under section 3.2; and 131 | 132 | iv) requires any subsequent distribution of the Program by any 133 | party to be under a license that satisfies the requirements 134 | of this section 3. 135 | 136 | 3.2 When the Program is Distributed as Source Code: 137 | 138 | a) it must be made available under this Agreement, or if the 139 | Program (i) is combined with other material in a separate file or 140 | files made available under a Secondary License, and (ii) the initial 141 | Contributor attached to the Source Code the notice described in 142 | Exhibit A of this Agreement, then the Program may be made available 143 | under the terms of such Secondary Licenses, and 144 | 145 | b) a copy of this Agreement must be included with each copy of 146 | the Program. 147 | 148 | 3.3 Contributors may not remove or alter any copyright, patent, 149 | trademark, attribution notices, disclaimers of warranty, or limitations 150 | of liability ("notices") contained within the Program from any copy of 151 | the Program which they Distribute, provided that Contributors may add 152 | their own appropriate notices. 153 | 154 | 4. COMMERCIAL DISTRIBUTION 155 | 156 | Commercial distributors of software may accept certain responsibilities 157 | with respect to end users, business partners and the like. While this 158 | license is intended to facilitate the commercial use of the Program, 159 | the Contributor who includes the Program in a commercial product 160 | offering should do so in a manner which does not create potential 161 | liability for other Contributors. Therefore, if a Contributor includes 162 | the Program in a commercial product offering, such Contributor 163 | ("Commercial Contributor") hereby agrees to defend and indemnify every 164 | other Contributor ("Indemnified Contributor") against any losses, 165 | damages and costs (collectively "Losses") arising from claims, lawsuits 166 | and other legal actions brought by a third party against the Indemnified 167 | Contributor to the extent caused by the acts or omissions of such 168 | Commercial Contributor in connection with its distribution of the Program 169 | in a commercial product offering. The obligations in this section do not 170 | apply to any claims or Losses relating to any actual or alleged 171 | intellectual property infringement. In order to qualify, an Indemnified 172 | Contributor must: a) promptly notify the Commercial Contributor in 173 | writing of such claim, and b) allow the Commercial Contributor to control, 174 | and cooperate with the Commercial Contributor in, the defense and any 175 | related settlement negotiations. The Indemnified Contributor may 176 | participate in any such claim at its own expense. 177 | 178 | For example, a Contributor might include the Program in a commercial 179 | product offering, Product X. That Contributor is then a Commercial 180 | Contributor. If that Commercial Contributor then makes performance 181 | claims, or offers warranties related to Product X, those performance 182 | claims and warranties are such Commercial Contributor's responsibility 183 | alone. Under this section, the Commercial Contributor would have to 184 | defend claims against the other Contributors related to those performance 185 | claims and warranties, and if a court requires any other Contributor to 186 | pay any damages as a result, the Commercial Contributor must pay 187 | those damages. 188 | 189 | 5. NO WARRANTY 190 | 191 | EXCEPT AS EXPRESSLY SET FORTH IN THIS AGREEMENT, AND TO THE EXTENT 192 | PERMITTED BY APPLICABLE LAW, THE PROGRAM IS PROVIDED ON AN "AS IS" 193 | BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, EITHER EXPRESS OR 194 | IMPLIED INCLUDING, WITHOUT LIMITATION, ANY WARRANTIES OR CONDITIONS OF 195 | TITLE, NON-INFRINGEMENT, MERCHANTABILITY OR FITNESS FOR A PARTICULAR 196 | PURPOSE. Each Recipient is solely responsible for determining the 197 | appropriateness of using and distributing the Program and assumes all 198 | risks associated with its exercise of rights under this Agreement, 199 | including but not limited to the risks and costs of program errors, 200 | compliance with applicable laws, damage to or loss of data, programs 201 | or equipment, and unavailability or interruption of operations. 202 | 203 | 6. DISCLAIMER OF LIABILITY 204 | 205 | EXCEPT AS EXPRESSLY SET FORTH IN THIS AGREEMENT, AND TO THE EXTENT 206 | PERMITTED BY APPLICABLE LAW, NEITHER RECIPIENT NOR ANY CONTRIBUTORS 207 | SHALL HAVE ANY LIABILITY FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 208 | EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING WITHOUT LIMITATION LOST 209 | PROFITS), HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 210 | CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 211 | ARISING IN ANY WAY OUT OF THE USE OR DISTRIBUTION OF THE PROGRAM OR THE 212 | EXERCISE OF ANY RIGHTS GRANTED HEREUNDER, EVEN IF ADVISED OF THE 213 | POSSIBILITY OF SUCH DAMAGES. 214 | 215 | 7. GENERAL 216 | 217 | If any provision of this Agreement is invalid or unenforceable under 218 | applicable law, it shall not affect the validity or enforceability of 219 | the remainder of the terms of this Agreement, and without further 220 | action by the parties hereto, such provision shall be reformed to the 221 | minimum extent necessary to make such provision valid and enforceable. 222 | 223 | If Recipient institutes patent litigation against any entity 224 | (including a cross-claim or counterclaim in a lawsuit) alleging that the 225 | Program itself (excluding combinations of the Program with other software 226 | or hardware) infringes such Recipient's patent(s), then such Recipient's 227 | rights granted under Section 2(b) shall terminate as of the date such 228 | litigation is filed. 229 | 230 | All Recipient's rights under this Agreement shall terminate if it 231 | fails to comply with any of the material terms or conditions of this 232 | Agreement and does not cure such failure in a reasonable period of 233 | time after becoming aware of such noncompliance. If all Recipient's 234 | rights under this Agreement terminate, Recipient agrees to cease use 235 | and distribution of the Program as soon as reasonably practicable. 236 | However, Recipient's obligations under this Agreement and any licenses 237 | granted by Recipient relating to the Program shall continue and survive. 238 | 239 | Everyone is permitted to copy and distribute copies of this Agreement, 240 | but in order to avoid inconsistency the Agreement is copyrighted and 241 | may only be modified in the following manner. The Agreement Steward 242 | reserves the right to publish new versions (including revisions) of 243 | this Agreement from time to time. No one other than the Agreement 244 | Steward has the right to modify this Agreement. The Eclipse Foundation 245 | is the initial Agreement Steward. The Eclipse Foundation may assign the 246 | responsibility to serve as the Agreement Steward to a suitable separate 247 | entity. Each new version of the Agreement will be given a distinguishing 248 | version number. The Program (including Contributions) may always be 249 | Distributed subject to the version of the Agreement under which it was 250 | received. In addition, after a new version of the Agreement is published, 251 | Contributor may elect to Distribute the Program (including its 252 | Contributions) under the new version. 253 | 254 | Except as expressly stated in Sections 2(a) and 2(b) above, Recipient 255 | receives no rights or licenses to the intellectual property of any 256 | Contributor under this Agreement, whether expressly, by implication, 257 | estoppel or otherwise. All rights in the Program not expressly granted 258 | under this Agreement are reserved. Nothing in this Agreement is intended 259 | to be enforceable by any entity that is not a Contributor or Recipient. 260 | No third-party beneficiary rights are created under this Agreement. 261 | 262 | Exhibit A - Form of Secondary Licenses Notice 263 | 264 | "This Source Code may also be made available under the following 265 | Secondary Licenses when the conditions for such availability set forth 266 | in the Eclipse Public License, v. 2.0 are satisfied: {name license(s), 267 | version(s), and exceptions or additional permissions here}." 268 | 269 | Simply including a copy of this Agreement, including this Exhibit A 270 | is not sufficient to license the Source Code under Secondary Licenses. 271 | 272 | If it is not possible or desirable to put the notice in a particular 273 | file, then You may include the notice in a location (such as a LICENSE 274 | file in a relevant directory) where a recipient would be likely to 275 | look for such a notice. 276 | 277 | You may add additional accurate notices of copyright ownership. 278 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # mikkmokk-proxy 2 | 3 | mikkmokk-proxy is an unobtrusive reverse proxy server that injects faults on the HTTP layer. 4 | 5 | Have you ever wanted to test how well your (backend|frontend) services handles 6 | * failed requests 7 | * duplicate requests 8 | * delayed requests 9 | 10 | mikkmokk-proxy is — literally — a gateway drug for resiliency testing on the HTTP layer. 11 | 12 | ## Overview 13 | 14 | ```mermaid 15 | sequenceDiagram 16 | client->>mikkmokk: POST 17 | mikkmokk->>mikkmokk: 💤delay before?💤 18 | mikkmokk->>client: 💥fail before?💥 19 | mikkmokk->>destination: POST 20 | mikkmokk->>destination: 💥duplicate POST?💥 21 | destination->>database: read and/or state change 22 | database->>destination: result 23 | destination->>mikkmokk: result 24 | mikkmokk->>mikkmokk: 💤delay after?💤 25 | mikkmokk->>client: result OR 💥fail after?💥 26 | ``` 27 | 28 | `client` here is anything that will normally access `destination` 29 | using the HTTP protocol, but goes via `mikkmokk` instead. 30 | 31 | mikkmokk can inject five different types of faults: 32 | * fail a request before the destination is reached 33 | * fail a request after the destination is reached 34 | * add a delay before accessing the destination 35 | * add a delay after accessing the destination 36 | * add a duplicate a request 37 | 38 | mikkmokk does fault injection based on a percentage chance. 39 | The scope for fault injection may be narrowed further by settings 40 | various matching criteria (URI, request method, header name/value pair, etc). 41 | 42 | The percentage chances, and related settings, can be 43 | set both statically and dynamically: 44 | * Using proxy headers `x-mikkmokk-...` when accessing the reverse proxy. 45 | * At runtime using the admin API, both for setting new defaults and for introducing one-off errors. 46 | * At startup time using environment variables. 47 | 48 | mikkmokk supports proxying to arbitrary URLs. 49 | 50 | ## Example usage 51 | 52 | #### Start the reverse proxy 53 | 54 | The following setup proxies to `http://example.com`: 55 | ```bash 56 | $ docker run --rm --name mikkmokk-proxy \ 57 | -e DESTINATION_URL=http://example.com \ 58 | -e PROXY_BIND=0.0.0.0 \ 59 | -e PROXY_PORT=8080 \ 60 | -e ADMIN_BIND=0.0.0.0 \ 61 | -e ADMIN_PORT=7070 \ 62 | -p 8080:8080 \ 63 | -p 7070:7070 \ 64 | docker.io/ivarref/mikkmokk-proxy:v0.1.63 65 | ``` 66 | 67 | There are two ports being exposed: 68 | * The reverse proxy on port 8080. 69 | * The admin server on port 7070. 70 | 71 | #### Issue a regular request 72 | 73 | ``` 74 | $ curl http://localhost:8080 75 | ... 76 |
This domain is for use in illustrative examples in documents. You may use this 78 | domain in literature without prior coordination or asking for permission.
79 | 80 | ... 81 | ``` 82 | This request succeeded because mikkmokk was not told to do any fault injection. 83 | 84 | #### Insert a failure before a request reaches the destination 85 | 86 | The header `x-mikkmokk-fail-before-percentage` can be used to simulate that 87 | the destination could not be reached. If present, it must be an int in the range 88 | \[0, 100\], i.e. it's the percentage chance that a request fails. 89 | The default value for this header is `0`. 90 | The value `100` means that the request will always fail. 91 | This can be used to test if clients are retrying or not. 92 | 93 | ``` 94 | $ curl -v -H 'x-mikkmokk-fail-before-percentage: 100' http://localhost:8080 95 | ... 96 | < HTTP/1.1 503 Service Unavailable 97 | < Content-Type: application/json 98 | ... 99 | {"error":"fail-before"} 100 | ``` 101 | 102 | The default HTTP status code for this is `503`, and may be changed 103 | using the header `x-mikkmokk-fail-before-code`. 104 | 105 | #### Insert a failure after a request has been processed by the destination 106 | 107 | The header `x-mikkmokk-fail-after-percentage` can be used to simulate that the 108 | destination has received and processed the request, but 109 | that the network between the proxy and the destination failed before 110 | the proxy received the response. Thus, the client will receive an incorrect response. 111 | If the client retries, will the backend handle a duplicate request? 112 | 113 | ``` 114 | $ curl -v -H 'x-mikkmokk-fail-after-percentage: 100' http://localhost:8080 115 | ... 116 | < HTTP/1.1 502 Bad Gateway 117 | < Content-Type: application/json 118 | ... 119 | {"error":"fail-after","destination-response-code":200} 120 | ``` 121 | 122 | The field `destination-response-code` states which HTTP status code the destination 123 | actually responded with. 124 | The default HTTP status code for this is `502`, and may be changed using the header `x-mikkmokk-fail-after-code`. 125 | 126 | #### Insert a duplicate request 127 | 128 | The header `x-mikkmokk-duplicate-percentage` instructs mikkmokk to make two identical, parallel requests. 129 | 130 | ``` 131 | $ curl -H 'x-mikkmokk-duplicate-percentage: 100' http://localhost:8080 132 | 133 | # In the mikkmokk logs you will see something like: 134 | > Duplicate request returned identical HTTP status code 200 for GET http://example.com/ 135 | ``` 136 | 137 | #### Only match a specific URI and/or request method 138 | 139 | ``` 140 | $ curl -H 'x-mikkmokk-match-uri: /something' \ 141 | -H 'x-mikkmokk-match-method: GET' \ 142 | -H 'x-mikkmokk-fail-before-percentage: 100' \ 143 | http://localhost:8080/something 144 | {"error":"fail-before"} 145 | ``` 146 | 147 | The default value of the `x-mikkmokk-match-uri` and `x-mikkmokk-match-method` headers is `*`, meaning that all URIs and all request methods will match. 148 | 149 | If you only want to match a given URI prefix, you may use the `x-mikkmokk-match-uri-starts-with` header. 150 | 151 | #### Only match a given header name/value pair 152 | 153 | ``` 154 | $ curl -H 'x-mikkmokk-match-header-name: x-some-header' \ 155 | -H 'x-mikkmokk-match-header-value: foobar' \ 156 | -H 'x-mikkmokk-fail-before-percentage: 100' \ 157 | http://localhost:8080/ 158 | ... request succeeds, header-name and -value did not match. 159 | 160 | $ curl -H 'x-mikkmokk-match-header-name: x-some-header' \ 161 | -H 'x-mikkmokk-match-header-value: foobar' \ 162 | -H 'x-mikkmokk-fail-before-percentage: 100' \ 163 | -H 'x-some-header: foobar' \ 164 | http://localhost:8080/ 165 | {"error":"fail-before"} 166 | ``` 167 | 168 | Here we see that the first request did not fail, and thus `x-mikkmokk-match-header-name` and 169 | `x-mikkmokk-header-value` did not match. 170 | 171 | On the second request it does fail however, and 172 | thus the header name-value pair did match. 173 | We explicitly set `x-some-header` ourselves. 174 | In a more real world setting it would be set by some gateway. 175 | 176 | #### Inserting delays 177 | 178 | Delays may be inserted using `x-mikkmokk-delay-before-percentage` and 179 | `x-mikkmokk-delay-before-ms`: 180 | 181 | ``` 182 | $ time curl -H 'x-mikkmokk-delay-before-percentage: 100' \ 183 | -H 'x-mikkmokk-delay-before-ms: 3000' \ 184 | http://localhost:8080 185 | ... 186 | real 0m3.252s 187 | ``` 188 | 189 | This delay will be inserted before the destination service is accessed. 190 | 191 | It's also possible to inject delays after the destination service has 192 | been accessed using `x-mikkmokk-delay-after-percentage` and 193 | `x-mikkmokk-delay-after-ms`. 194 | 195 | ### Use the admin API to introduce one-off errors 196 | 197 | Let's say that you want to test how a frontend handles a failed request, 198 | but you do not want edit the source code of the frontend. You also 199 | do not want to create any unnecessary errors. 200 | 201 | You can use the admin API for one-off errors for these tasks: 202 | 203 | ``` 204 | # Notice the port 7070 here, which is where we exposed the admin 205 | # API earlier: 206 | $ curl -XPOST -H 'x-mikkmokk-fail-before-percentage: 100' \ 207 | http://localhost:7070/api/v1/one-off 208 | {"service":"mikkmokk","message":"Added one-off"} 209 | 210 | # The next request now fails: 211 | $ curl http://localhost:8080 212 | {"error":"fail-before"} 213 | 214 | # The request after succeeds: 215 | $ curl http://localhost:8080 216 | ...