/*
|
Copyright (c) 2012-2015, Pierre-Olivier Latour
|
All rights reserved.
|
|
Redistribution and use in source and binary forms, with or without
|
modification, are permitted provided that the following conditions are met:
|
* Redistributions of source code must retain the above copyright
|
notice, this list of conditions and the following disclaimer.
|
* Redistributions in binary form must reproduce the above copyright
|
notice, this list of conditions and the following disclaimer in the
|
documentation and/or other materials provided with the distribution.
|
* The name of Pierre-Olivier Latour may not be used to endorse
|
or promote products derived from this software without specific
|
prior written permission.
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
DISCLAIMED. IN NO EVENT SHALL PIERRE-OLIVIER LATOUR BE LIABLE FOR ANY
|
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
*/
|
|
#import <Foundation/Foundation.h>
|
|
NS_ASSUME_NONNULL_BEGIN
|
|
/**
|
* The GCDWebServerBodyReaderCompletionBlock is passed by GCDWebServer to the
|
* GCDWebServerBodyReader object when reading data from it asynchronously.
|
*/
|
typedef void (^GCDWebServerBodyReaderCompletionBlock)(NSData* data, NSError* _Nullable error);
|
|
/**
|
* This protocol is used by the GCDWebServerConnection to communicate with
|
* the GCDWebServerResponse and read the HTTP body data to send.
|
*
|
* Note that multiple GCDWebServerBodyReader objects can be chained together
|
* internally e.g. to automatically apply gzip encoding to the content before
|
* passing it on to the GCDWebServerResponse.
|
*
|
* @warning These methods can be called on any GCD thread.
|
*/
|
@protocol GCDWebServerBodyReader <NSObject>
|
|
@required
|
|
/**
|
* This method is called before any body data is sent.
|
*
|
* It should return YES on success or NO on failure and set the "error" argument
|
* which is guaranteed to be non-NULL.
|
*/
|
- (BOOL)open:(NSError**)error;
|
|
/**
|
* This method is called whenever body data is sent.
|
*
|
* It should return a non-empty NSData if there is body data available,
|
* or an empty NSData there is no more body data, or nil on error and set
|
* the "error" argument which is guaranteed to be non-NULL.
|
*/
|
- (nullable NSData*)readData:(NSError**)error;
|
|
/**
|
* This method is called after all body data has been sent.
|
*/
|
- (void)close;
|
|
@optional
|
|
/**
|
* If this method is implemented, it will be preferred over -readData:.
|
*
|
* It must call the passed block when data is available, passing a non-empty
|
* NSData if there is body data available, or an empty NSData there is no more
|
* body data, or nil on error and pass an NSError along.
|
*/
|
- (void)asyncReadDataWithCompletion:(GCDWebServerBodyReaderCompletionBlock)block;
|
|
@end
|
|
/**
|
* The GCDWebServerResponse class is used to wrap a single HTTP response.
|
* It is instantiated by the handler of the GCDWebServer that handled the request.
|
* If a body is present, the methods from the GCDWebServerBodyReader protocol
|
* will be called by the GCDWebServerConnection to send it.
|
*
|
* The default implementation of the GCDWebServerBodyReader protocol
|
* on the class simply returns an empty body.
|
*
|
* @warning GCDWebServerResponse instances can be created and used on any GCD thread.
|
*/
|
@interface GCDWebServerResponse : NSObject <GCDWebServerBodyReader>
|
|
/**
|
* Sets the content type for the body of the response.
|
*
|
* The default value is nil i.e. the response has no body.
|
*
|
* @warning This property must be set if a body is present.
|
*/
|
@property(nonatomic, copy, nullable) NSString* contentType;
|
|
/**
|
* Sets the content length for the body of the response. If a body is present
|
* but this property is set to "NSUIntegerMax", this means the length of the body
|
* cannot be known ahead of time. Chunked transfer encoding will be
|
* automatically enabled by the GCDWebServerConnection to comply with HTTP/1.1
|
* specifications.
|
*
|
* The default value is "NSUIntegerMax" i.e. the response has no body or its length
|
* is undefined.
|
*/
|
@property(nonatomic) NSUInteger contentLength;
|
|
/**
|
* Sets the HTTP status code for the response.
|
*
|
* The default value is 200 i.e. "OK".
|
*/
|
@property(nonatomic) NSInteger statusCode;
|
|
/**
|
* Sets the caching hint for the response using the "Cache-Control" header.
|
* This value is expressed in seconds.
|
*
|
* The default value is 0 i.e. "no-cache".
|
*/
|
@property(nonatomic) NSUInteger cacheControlMaxAge;
|
|
/**
|
* Sets the last modified date for the response using the "Last-Modified" header.
|
*
|
* The default value is nil.
|
*/
|
@property(nonatomic, nullable) NSDate* lastModifiedDate;
|
|
/**
|
* Sets the ETag for the response using the "ETag" header.
|
*
|
* The default value is nil.
|
*/
|
@property(nonatomic, copy, nullable) NSString* eTag;
|
|
/**
|
* Enables gzip encoding for the response body.
|
*
|
* The default value is NO.
|
*
|
* @warning Enabling gzip encoding will remove any "Content-Length" header
|
* since the length of the body is not known anymore. The client will still
|
* be able to determine the body length when connection is closed per
|
* HTTP/1.1 specifications.
|
*/
|
@property(nonatomic, getter=isGZipContentEncodingEnabled) BOOL gzipContentEncodingEnabled;
|
|
/**
|
* Creates an empty response.
|
*/
|
+ (instancetype)response;
|
|
/**
|
* This method is the designated initializer for the class.
|
*/
|
- (instancetype)init;
|
|
/**
|
* Sets an additional HTTP header on the response.
|
* Pass a nil value to remove an additional header.
|
*
|
* @warning Do not attempt to override the primary headers used
|
* by GCDWebServerResponse like "Content-Type", "ETag", etc...
|
*/
|
- (void)setValue:(nullable NSString*)value forAdditionalHeader:(NSString*)header;
|
|
/**
|
* Convenience method that checks if the contentType property is defined.
|
*/
|
- (BOOL)hasBody;
|
|
@end
|
|
@interface GCDWebServerResponse (Extensions)
|
|
/**
|
* Creates a empty response with a specific HTTP status code.
|
*/
|
+ (instancetype)responseWithStatusCode:(NSInteger)statusCode;
|
|
/**
|
* Creates an HTTP redirect response to a new URL.
|
*/
|
+ (instancetype)responseWithRedirect:(NSURL*)location permanent:(BOOL)permanent;
|
|
/**
|
* Initializes an empty response with a specific HTTP status code.
|
*/
|
- (instancetype)initWithStatusCode:(NSInteger)statusCode;
|
|
/**
|
* Initializes an HTTP redirect response to a new URL.
|
*/
|
- (instancetype)initWithRedirect:(NSURL*)location permanent:(BOOL)permanent;
|
|
@end
|
|
NS_ASSUME_NONNULL_END
|