From 167d08babaa21eec77f77b28b6145c31e8aacb13 Mon Sep 17 00:00:00 2001 From: "Oscar S. Silva" Date: Wed, 13 Nov 2024 13:56:52 +0100 Subject: [PATCH] request map returns `:headers` map with lower-case string keys --- src/ring/util/http_response.clj | 16 ++-- test/ring/util/http_response_test.clj | 120 +++++++++++++++++++++++--- 2 files changed, 118 insertions(+), 18 deletions(-) diff --git a/src/ring/util/http_response.clj b/src/ring/util/http_response.clj index b648a4c..8203156 100644 --- a/src/ring/util/http_response.clj +++ b/src/ring/util/http_response.clj @@ -50,7 +50,7 @@ ([url] (created url nil)) ([url body] {:status 201 - :headers {"Location" url} + :headers {"location" url} :body body})) (defn accepted @@ -128,7 +128,7 @@ There are multiple options for the resource that the client may follow." ([url] {:status 300 - :headers {"Location" url} + :headers {"location" url} :body ""})) (defn moved-permanently @@ -136,7 +136,7 @@ This and all future requests should be directed to the given URI." ([url] {:status 301 - :headers {"Location" url} + :headers {"location" url} :body ""})) (defn found @@ -144,7 +144,7 @@ The resource was found but at a different URI." ([url] {:status 302 - :headers {"Location" url} + :headers {"location" url} :body ""})) (defn see-other @@ -152,7 +152,7 @@ The response to the request can be found under another URI using a GET method." ([url] {:status 303 - :headers {"Location" url} + :headers {"location" url} :body ""})) (defn not-modified @@ -168,7 +168,7 @@ This single request is to be repeated via the proxy given by the Location field." ([url] {:status 305 - :headers {"Location" url} + :headers {"location" url} :body ""})) (defn temporary-redirect @@ -176,7 +176,7 @@ The request should be repeated with another URI but future requests can still use the original URI." ([url] {:status 307 - :headers {"Location" url} + :headers {"location" url} :body ""})) (defn permanent-redirect @@ -184,7 +184,7 @@ The request and all future requests should be repeated using another URI." ([url] {:status 308 - :headers {"Location" url} + :headers {"location" url} :body ""})) (defn bad-request diff --git a/test/ring/util/http_response_test.clj b/test/ring/util/http_response_test.clj index ca31863..8e1273d 100644 --- a/test/ring/util/http_response_test.clj +++ b/test/ring/util/http_response_test.clj @@ -1,6 +1,8 @@ (ns ring.util.http-response-test (:require [clojure.test :refer :all] - [ring.util.http-response :refer :all])) + [ring.util.http-response :refer :all] + [clojure.string :as str] + [clojure.repl :refer [demunge]])) (deftest http-responses-test @@ -11,8 +13,8 @@ (testing "Success" (is (= {:status 200 :headers {} :body "body"} (ok "body"))) - (is (= {:status 201 :headers {"Location" "/url"} :body nil} (created "/url"))) - (is (= {:status 201 :headers {"Location" "/url"} :body "body"} (created "/url" "body"))) + (is (= {:status 201 :headers {"location" "/url"} :body nil} (created "/url"))) + (is (= {:status 201 :headers {"location" "/url"} :body "body"} (created "/url" "body"))) (is (= {:status 202 :headers {} :body "body"} (accepted "body"))) (is (= {:status 203 :headers {} :body "body"} (non-authoritative-information "body"))) (is (= {:status 204 :headers {} :body ""} (no-content))) @@ -23,14 +25,14 @@ (is (= {:status 226 :headers {} :body "body"} (im-used "body")))) (testing "Redirection" - (is (= {:status 300 :headers {"Location" "/url"} :body ""} (multiple-choices "/url"))) - (is (= {:status 301 :headers {"Location" "/url"} :body ""} (moved-permanently "/url"))) - (is (= {:status 302 :headers {"Location" "/url"} :body ""} (found "/url"))) - (is (= {:status 303 :headers {"Location" "/url"} :body ""} (see-other "/url"))) + (is (= {:status 300 :headers {"location" "/url"} :body ""} (multiple-choices "/url"))) + (is (= {:status 301 :headers {"location" "/url"} :body ""} (moved-permanently "/url"))) + (is (= {:status 302 :headers {"location" "/url"} :body ""} (found "/url"))) + (is (= {:status 303 :headers {"location" "/url"} :body ""} (see-other "/url"))) (is (= {:status 304 :headers {} :body ""} (not-modified))) - (is (= {:status 305 :headers {"Location" "/url"} :body ""} (use-proxy "/url"))) - (is (= {:status 307 :headers {"Location" "/url"} :body ""} (temporary-redirect "/url"))) - (is (= {:status 308 :headers {"Location" "/url"} :body ""} (permanent-redirect "/url")))) + (is (= {:status 305 :headers {"location" "/url"} :body ""} (use-proxy "/url"))) + (is (= {:status 307 :headers {"location" "/url"} :body ""} (temporary-redirect "/url"))) + (is (= {:status 308 :headers {"location" "/url"} :body ""} (permanent-redirect "/url")))) (testing "ClientError" (is (= {:status 400 :headers {} :body "body"} (bad-request "body"))) @@ -81,6 +83,104 @@ (is (= {:status 598 :headers {} :body "body"} (network-read-timeout "body"))) (is (= {:status 599 :headers {} :body "body"} (network-connect-timeout "body"))))) + +;; Functions that can be called with 1 argument +(def one-args-response-fns + [multiple-choices + moved-permanently + found + see-other + use-proxy + temporary-redirect + permanent-redirect]) + +;; Functions that can be called with 0 arguments +(def zero-arg-response-fns + [continue + switching-protocols + processing + ok + created + accepted + non-authoritative-information + no-content + reset-content + partial-content + multi-status + already-reported + im-used + not-modified + bad-request + unauthorized + payment-required + forbidden + not-found + method-not-allowed + not-acceptable + proxy-authentication-required + request-timeout + conflict + gone + length-required + precondition-failed + request-entity-too-large + request-uri-too-long + unsupported-media-type + requested-range-not-satisfiable + expectation-failed + im-a-teapot + enhance-your-calm + unprocessable-entity + locked + failed-dependency + unordered-collection + upgrade-required + precondition-required + too-many-requests + request-header-fields-too-large + retry-with + blocked-by-windows-parental-controls + unavailable-for-legal-reasons + internal-server-error + not-implemented + bad-gateway + service-unavailable + gateway-timeout + http-version-not-supported + variant-also-negotiates + insufficient-storage + loop-detected + bandwidth-limit-exceeded + not-extended + network-authentication-required + network-read-timeout + network-connect-timeout]) + +(defn- pretty-demunge [f] + (-> (str f) demunge (str/split #"@") first)) + +;; https://github.com/ring-clojure/ring/blob/master/SPEC.md +(deftest ring-spec-test + (testing "`:headers` is a clojure map of lowercased header name strings to corresponding header value strings" + ;; for functions that have a one-arity definition + (doseq [f one-args-response-fns + :let [request-map (f "foo") + headers (:headers request-map)]] + (testing (pretty-demunge f) + (is (map? headers)) + (is (every? (fn [header-name] + (= (str/lower-case header-name) header-name)) + (keys headers)))) + ;; for functions that have a zero-arity definition + (doseq [f zero-arg-response-fns + :let [request-map (f) + headers (:headers request-map)]] + (testing (pretty-demunge f) + (is (map? headers)) + (is (every? (fn [header-name] + (= (str/lower-case header-name) header-name)) + (keys headers)))))))) + (declare slingshots?) (defmethod assert-expr 'slingshots? [msg form] (let [expected (second form)