Skip to content

Commit

Permalink
Merge pull request #61 from ba-st/no-store
Browse files Browse the repository at this point in the history
Add support for `Cache-Control: no-store` directive
  • Loading branch information
gcotelli authored Jul 4, 2023
2 parents 41ac157 + d4c0af7 commit 2db2ca1
Show file tree
Hide file tree
Showing 9 changed files with 175 additions and 63 deletions.
4 changes: 3 additions & 1 deletion docs/reference/API-Client.md
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,8 @@ scenarios:
- When using any of the `GET`-related methods, if an entity tag was previously
saved its value is used in an `If-None-Match` header. Giving the server the
chance to respond with `204/Not modified` and in that case, the previously
saved body is used as the response body.
saved body is used as the response body; unless a `no-store` caching directive
is in place.

## Caching

Expand Down Expand Up @@ -145,3 +146,4 @@ The following caching headers are supported:
- `Cache-Control`
- `Max-Age`
- `no-cache`
- `no-store`
16 changes: 6 additions & 10 deletions source/Superluminal-RESTfulAPI-Tests/ExpiringCacheTest.class.st
Original file line number Diff line number Diff line change
Expand Up @@ -37,15 +37,11 @@ ExpiringCacheTest >> obtain: answer cachedFor: aDuration [
{ #category : #private }
ExpiringCacheTest >> resourceFor: answer expiringIn: aDuration [

| headers response |

headers := Dictionary new
at: 'Date' put: ( ZnUtils httpDate: currentDateTime );
at: 'Cache-Control' put: ( Array with: ( 'Max-Age=<1p>' expandMacrosWith: aDuration asSeconds ) );
yourself.
response := ZnResponse noContent
headers: headers;
yourself.
| response |

response := ZnResponse noContent.
response headers at: 'Date' put: ( ZnUtils httpDate: currentDateTime ).
response addCachingDirective: ( 'Max-Age=<1p>' expandMacrosWith: aDuration asSeconds ).
^ ExpiringResource for: answer controlledBy: response
]

Expand All @@ -55,7 +51,7 @@ ExpiringCacheTest >> setUp [
super setUp.
currentDateTime := DateAndTime now truncated.
currentTimeProvider := [ currentDateTime ].
location := UUID new asString.
location := 'answers' asUrl / UUID new asString.
self setUpExpiringCache
]

Expand Down
28 changes: 28 additions & 0 deletions source/Superluminal-RESTfulAPI-Tests/ExpiringResourceTest.class.st
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,34 @@ ExpiringResourceTest >> setUp [
currentDateTime := DateAndTime now truncated
]

{ #category : #tests }
ExpiringResourceTest >> testCanBeStored [

| resource |

self
assert: ( self resourceOriginatedAt: currentDateTime expiresContaining: 'not-a-date' ) canBeStored;
assert: ( self resourceOriginatedAt: currentDateTime expiringIn: 0 seconds ) canBeStored;
assert: ( self resourceWithNoCacheDirectiveOriginatedAt: currentDateTime ) canBeStored;
assert:
( self resourceWithNoCacheDirectiveOriginatedAt: currentDateTime expiringIn: 1 second )
canBeStored.


resource := self expiringResourceControlledBy: ( ( self responseWithDateHeader: currentDateTime )
addCachingDirective: 'no-store';
yourself ).

self deny: resource canBeStored.

resource := self expiringResourceControlledBy: ( ( self responseWithDateHeader: currentDateTime )
addCachingDirective: 'no-cache';
addCachingDirective: 'no-store';
yourself ).

self deny: resource canBeStored
]

{ #category : #tests }
ExpiringResourceTest >> testExpiredWhenExpiresSetToInvalidDate [
"A cache recipient MUST interpret invalid date formats, especially the
Expand Down
120 changes: 89 additions & 31 deletions source/Superluminal-RESTfulAPI-Tests/RESTfulAPIClientTest.class.st
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ RESTfulAPIClientTest >> apiClient [
^ self subclassResponsibility
]

{ #category : #tests }
{ #category : #private }
RESTfulAPIClientTest >> jsonOkResponse [

^ self jsonOkResponseWith: #(1 2 3)
Expand All @@ -32,7 +32,7 @@ RESTfulAPIClientTest >> location [
^ 'http://localhost' asAbsoluteUrl + resourceIdentifier
]

{ #category : #tests }
{ #category : #private }
RESTfulAPIClientTest >> notFoundResponse [

^ ZnResponse notFound: self location
Expand All @@ -53,7 +53,7 @@ RESTfulAPIClientTest >> tearDown [
super tearDown
]

{ #category : #tests }
{ #category : #'tests - DELETE' }
RESTfulAPIClientTest >> testDeleteAcceptingWithSuccessfulResponseDo [

| wasSuccessfull |
Expand All @@ -68,14 +68,14 @@ RESTfulAPIClientTest >> testDeleteAcceptingWithSuccessfulResponseDo [
self assert: wasSuccessfull
]

{ #category : #tests }
{ #category : #'tests - DELETE' }
RESTfulAPIClientTest >> testDeleteAtSuccess [

self configureHttpClientToRespondWith: ZnResponse noContent.
self shouldnt: [ apiClient deleteAt: self location ] raise: HTTPClientError
]

{ #category : #tests }
{ #category : #'tests - DELETE' }
RESTfulAPIClientTest >> testDeleteAtSuccessWhenCached [

| etagIsSet |
Expand All @@ -96,7 +96,7 @@ RESTfulAPIClientTest >> testDeleteAtSuccessWhenCached [
self assert: etagIsSet
]

{ #category : #tests }
{ #category : #'tests - DELETE' }
RESTfulAPIClientTest >> testDeleteNotFound [

self configureHttpClientToRespondWith: self notFoundResponse.
Expand All @@ -110,7 +110,7 @@ RESTfulAPIClientTest >> testDeleteNotFound [
raise: HTTPClientError notFound
]

{ #category : #tests }
{ #category : #'tests - GET' }
RESTfulAPIClientTest >> testGetAcceptingWithSuccessfulResponseDo [

self configureHttpClientToRespondWith: self jsonOkResponse.
Expand All @@ -122,29 +122,61 @@ RESTfulAPIClientTest >> testGetAcceptingWithSuccessfulResponseDo [
[ :responseContents | self withJsonFrom: responseContents do: [ :json | self assert: json equals: #(1 2 3) ] ]
]

{ #category : #tests }
{ #category : #'tests - GET' }
RESTfulAPIClientTest >> testGetCached [

self configureHttpClientToRespondWith: ( ( self jsonOkResponseWith: #( 1 2 3 ) )
addCachingDirective: 'Max-Age=60';
yourself ).

apiClient get: self location withSuccessfulResponseDo: [ :responseContents |
self withJsonFrom: responseContents do: [ :json | self assert: json equals: #( 1 2 3 ) ] ].

self configureHttpClientToRespondWith: self notFoundResponse.

apiClient get: self location withSuccessfulResponseDo: [ :responseContents |
self withJsonFrom: responseContents do: [ :json | self assert: json equals: #( 1 2 3 ) ] ]
]

{ #category : #'tests - GET' }
RESTfulAPIClientTest >> testGetDoNotStoreResponseInCache [

self configureHttpClientToRespondWith: ( ( self jsonOkResponseWith: #( 1 2 3 ) )
addCachingDirective: 'no-store';
yourself ).

apiClient get: self location withSuccessfulResponseDo: [ :responseContents |
self withJsonFrom: responseContents do: [ :json | self assert: json equals: #( 1 2 3 ) ] ].

self configureHttpClientToRespondWith: self notFoundResponse.

self
configureHttpClientToRespondWith:
( ( self jsonOkResponseWith: #(1 2 3) )
addCachingDirective: 'Max-Age=60';
yourself ).
should: [
apiClient get: self location withSuccessfulResponseDo: [ :responseContents | self fail ] ]
raise: HTTPClientError notFound
]

apiClient
get: self location
withSuccessfulResponseDo:
[ :responseContents | self withJsonFrom: responseContents do: [ :json | self assert: json equals: #(1 2 3) ] ].
{ #category : #'tests - GET' }
RESTfulAPIClientTest >> testGetIgnoreETagsWhenDoNotStoreCachingPolicyIsInPlace [

self configureHttpClientToRespondWith: ( ZnResponse notFound: self location ).
self configureHttpClientToRespondWith: ( ( self jsonOkResponseWith: #( 1 2 3 ) )
setEntityTag: '"123"';
addCachingDirective: 'no-store';
yourself ).

apiClient
get: self location
withSuccessfulResponseDo:
[ :responseContents | self withJsonFrom: responseContents do: [ :json | self assert: json equals: #(1 2 3) ] ]
apiClient get: self location withSuccessfulResponseDo: [ :responseContents |
self withJsonFrom: responseContents do: [ :json | self assert: json equals: #( 1 2 3 ) ] ].

self configureHttpClientToRespondWith: ( ( self jsonOkResponseWith: #( 1 2 3 ) )
setEntityTag: '"123"';
addCachingDirective: 'no-store';
yourself ).

apiClient get: self location withSuccessfulResponseDo: [ :responseContents |
self withJsonFrom: responseContents do: [ :json | self assert: json equals: #( 1 2 3 ) ] ]
]

{ #category : #tests }
{ #category : #'tests - GET' }
RESTfulAPIClientTest >> testGetNotFound [

self configureHttpClientToRespondWith: self notFoundResponse.
Expand All @@ -154,7 +186,33 @@ RESTfulAPIClientTest >> testGetNotFound [
raise: HTTPClientError notFound
]

{ #category : #tests }
{ #category : #'tests - GET' }
RESTfulAPIClientTest >> testGetUsingETagInASecondInvocation [

| ifNoneMatchHeaderWasSet |

self configureHttpClientToRespondWith: ( ( self jsonOkResponseWith: #( 1 2 3 ) )
setEntityTag: '"123"';
yourself ).

apiClient get: self location withSuccessfulResponseDo: [ :responseContents |
self withJsonFrom: responseContents do: [ :json | self assert: json equals: #( 1 2 3 ) ] ].

self configureHttpClientToRespondWith: ZnResponse notModified.

ifNoneMatchHeaderWasSet := false.
self httpClient whenSend: #setIfNoneMatchTo: evaluate: [ :etag |
self assert: etag equals: '"123"' asEntityTag.
ifNoneMatchHeaderWasSet := true
].

apiClient get: self location withSuccessfulResponseDo: [ :responseContents |
self withJsonFrom: responseContents do: [ :json | self assert: json equals: #( 1 2 3 ) ] ].

self assert: ifNoneMatchHeaderWasSet
]

{ #category : #'tests - GET' }
RESTfulAPIClientTest >> testGetWithSuccessfulResponseDo [

self configureHttpClientToRespondWith: self jsonOkResponse.
Expand All @@ -165,7 +223,7 @@ RESTfulAPIClientTest >> testGetWithSuccessfulResponseDo [
[ :responseContents | self withJsonFrom: responseContents do: [ :json | self assert: json equals: #(1 2 3) ] ]
]

{ #category : #tests }
{ #category : #'tests - PATCH' }
RESTfulAPIClientTest >> testPatchAtNoContent [

self configureHttpClientToRespondWith: ZnResponse noContent.
Expand All @@ -175,7 +233,7 @@ RESTfulAPIClientTest >> testPatchAtNoContent [
withSuccessfulResponseDo: [ :responseContents | self assert: responseContents isNil ]
]

{ #category : #tests }
{ #category : #'tests - PATCH' }
RESTfulAPIClientTest >> testPatchAtNotFound [

self configureHttpClientToRespondWith: self notFoundResponse.
Expand All @@ -189,7 +247,7 @@ RESTfulAPIClientTest >> testPatchAtNotFound [
withMessageText: 'Cannot complete update'
]

{ #category : #tests }
{ #category : #'tests - PATCH' }
RESTfulAPIClientTest >> testPatchAtWithSuccessfulResponseDo [

self configureHttpClientToRespondWith: self jsonOkResponse.
Expand All @@ -201,7 +259,7 @@ RESTfulAPIClientTest >> testPatchAtWithSuccessfulResponseDo [
[ :responseContents | self withJsonFrom: responseContents do: [ :json | self assert: json equals: #(1 2 3) ] ]
]

{ #category : #tests }
{ #category : #'tests - POST' }
RESTfulAPIClientTest >> testPostBadRequest [

self
Expand All @@ -216,7 +274,7 @@ RESTfulAPIClientTest >> testPostBadRequest [
withMessageText: 'Cannot complete the request'
]

{ #category : #tests }
{ #category : #'tests - POST' }
RESTfulAPIClientTest >> testPostWithSuccessfulResponseDo [

self configureHttpClientToRespondWith: ( ZnResponse created: self location ).
Expand All @@ -226,7 +284,7 @@ RESTfulAPIClientTest >> testPostWithSuccessfulResponseDo [
withSuccessfulResponseDo: [ :responseContents | self assert: ( responseContents beginsWith: 'Created' ) ]
]

{ #category : #tests }
{ #category : #'tests - PUT' }
RESTfulAPIClientTest >> testPutAt [

| response |
Expand All @@ -238,7 +296,7 @@ RESTfulAPIClientTest >> testPutAt [
self withJsonFrom: response contents do: [ :json | self assert: json equals: #(1 2 3) ]
]

{ #category : #tests }
{ #category : #'tests - PUT' }
RESTfulAPIClientTest >> testPutAtNoContent [

| response |
Expand All @@ -249,7 +307,7 @@ RESTfulAPIClientTest >> testPutAtNoContent [
self assert: response isNoContent
]

{ #category : #tests }
{ #category : #'tests - PUT' }
RESTfulAPIClientTest >> testPutAtNotFound [

self configureHttpClientToRespondWith: self notFoundResponse.
Expand Down
16 changes: 10 additions & 6 deletions source/Superluminal-RESTfulAPI/ExpiringCache.class.st
Original file line number Diff line number Diff line change
Expand Up @@ -51,12 +51,16 @@ ExpiringCache >> withResourceAt: aBuiltKey obtainedUsing: aGetBlock do: aProcess

| resource |

[ resource := expiringResources get: aBuiltKey ]
on: KeyNotFound
do: [ :signal |
resource := aGetBlock value.
expiringResources store: resource at: aBuiltKey
].
resource := [ expiringResources get: aBuiltKey ]
on: KeyNotFound
do: [ :notFound |
| obtainedResource |

obtainedResource := aGetBlock value.
obtainedResource canBeStored then: [
expiringResources store: obtainedResource at: aBuiltKey ].
notFound return: obtainedResource
].

^ aProcessBlock value: resource contents
]
6 changes: 6 additions & 0 deletions source/Superluminal-RESTfulAPI/ExpiringResource.class.st
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,12 @@ ExpiringResource class >> for: aResource controlledBy: aResponse [
^ self new initializeFor: aResource controlledBy: aResponse
]

{ #category : #testing }
ExpiringResource >> canBeStored [

^ response canBeStoredInCache
]

{ #category : #accessing }
ExpiringResource >> contents [

Expand Down
Loading

0 comments on commit 2db2ca1

Please sign in to comment.