GCDWebServerPrivate.h 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. /*
  2. Copyright (c) 2012-2015, Pierre-Olivier Latour
  3. All rights reserved.
  4. Redistribution and use in source and binary forms, with or without
  5. modification, are permitted provided that the following conditions are met:
  6. * Redistributions of source code must retain the above copyright
  7. notice, this list of conditions and the following disclaimer.
  8. * Redistributions in binary form must reproduce the above copyright
  9. notice, this list of conditions and the following disclaimer in the
  10. documentation and/or other materials provided with the distribution.
  11. * The name of Pierre-Olivier Latour may not be used to endorse
  12. or promote products derived from this software without specific
  13. prior written permission.
  14. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
  15. ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  16. WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  17. DISCLAIMED. IN NO EVENT SHALL PIERRE-OLIVIER LATOUR BE LIABLE FOR ANY
  18. DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  19. (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  20. LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  21. ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  22. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  23. SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  24. */
  25. #import <os/object.h>
  26. #import <sys/socket.h>
  27. /**
  28. * All GCDWebServer headers.
  29. */
  30. #import "GCDWebServerHTTPStatusCodes.h"
  31. #import "GCDWebServerFunctions.h"
  32. #import "GCDWebServer.h"
  33. #import "GCDWebServerConnection.h"
  34. #import "GCDWebServerDataRequest.h"
  35. #import "GCDWebServerFileRequest.h"
  36. #import "GCDWebServerMultiPartFormRequest.h"
  37. #import "GCDWebServerURLEncodedFormRequest.h"
  38. #import "GCDWebServerDataResponse.h"
  39. #import "GCDWebServerErrorResponse.h"
  40. #import "GCDWebServerFileResponse.h"
  41. #import "GCDWebServerStreamedResponse.h"
  42. /**
  43. * Check if a custom logging facility should be used instead.
  44. */
  45. #if defined(__GCDWEBSERVER_LOGGING_HEADER__)
  46. #define __GCDWEBSERVER_LOGGING_FACILITY_CUSTOM__
  47. #import __GCDWEBSERVER_LOGGING_HEADER__
  48. /**
  49. * Automatically detect if XLFacility is available and if so use it as a
  50. * logging facility.
  51. */
  52. #elif defined(__has_include) && __has_include("XLFacilityMacros.h")
  53. #define __GCDWEBSERVER_LOGGING_FACILITY_XLFACILITY__
  54. #undef XLOG_TAG
  55. #define XLOG_TAG @"gcdwebserver.internal"
  56. #import "XLFacilityMacros.h"
  57. #define GWS_LOG_DEBUG(...) XLOG_DEBUG(__VA_ARGS__)
  58. #define GWS_LOG_VERBOSE(...) XLOG_VERBOSE(__VA_ARGS__)
  59. #define GWS_LOG_INFO(...) XLOG_INFO(__VA_ARGS__)
  60. #define GWS_LOG_WARNING(...) XLOG_WARNING(__VA_ARGS__)
  61. #define GWS_LOG_ERROR(...) XLOG_ERROR(__VA_ARGS__)
  62. #define GWS_DCHECK(__CONDITION__) XLOG_DEBUG_CHECK(__CONDITION__)
  63. #define GWS_DNOT_REACHED() XLOG_DEBUG_UNREACHABLE()
  64. /**
  65. * Automatically detect if CocoaLumberJack is available and if so use
  66. * it as a logging facility.
  67. */
  68. #elif defined(__has_include) && __has_include("CocoaLumberjack/CocoaLumberjack.h")
  69. #import <CocoaLumberjack/CocoaLumberjack.h>
  70. #define __GCDWEBSERVER_LOGGING_FACILITY_COCOALUMBERJACK__
  71. #undef LOG_LEVEL_DEF
  72. #define LOG_LEVEL_DEF GCDWebServerLogLevel
  73. extern DDLogLevel GCDWebServerLogLevel;
  74. #define GWS_LOG_DEBUG(...) DDLogDebug(__VA_ARGS__)
  75. #define GWS_LOG_VERBOSE(...) DDLogVerbose(__VA_ARGS__)
  76. #define GWS_LOG_INFO(...) DDLogInfo(__VA_ARGS__)
  77. #define GWS_LOG_WARNING(...) DDLogWarn(__VA_ARGS__)
  78. #define GWS_LOG_ERROR(...) DDLogError(__VA_ARGS__)
  79. /**
  80. * If all of the above fail, then use GCDWebServer built-in
  81. * logging facility.
  82. */
  83. #else
  84. #define __GCDWEBSERVER_LOGGING_FACILITY_BUILTIN__
  85. typedef NS_ENUM(int, GCDWebServerLoggingLevel) {
  86. kGCDWebServerLoggingLevel_Debug = 0,
  87. kGCDWebServerLoggingLevel_Verbose,
  88. kGCDWebServerLoggingLevel_Info,
  89. kGCDWebServerLoggingLevel_Warning,
  90. kGCDWebServerLoggingLevel_Error
  91. };
  92. extern GCDWebServerLoggingLevel GCDWebServerLogLevel;
  93. extern void GCDWebServerLogMessage(GCDWebServerLoggingLevel level, NSString* format, ...) NS_FORMAT_FUNCTION(2, 3);
  94. #if DEBUG
  95. #define GWS_LOG_DEBUG(...) do { if (GCDWebServerLogLevel <= kGCDWebServerLoggingLevel_Debug) GCDWebServerLogMessage(kGCDWebServerLoggingLevel_Debug, __VA_ARGS__); } while (0)
  96. #else
  97. #define GWS_LOG_DEBUG(...)
  98. #endif
  99. #define GWS_LOG_VERBOSE(...) do { if (GCDWebServerLogLevel <= kGCDWebServerLoggingLevel_Verbose) GCDWebServerLogMessage(kGCDWebServerLoggingLevel_Verbose, __VA_ARGS__); } while (0)
  100. #define GWS_LOG_INFO(...) do { if (GCDWebServerLogLevel <= kGCDWebServerLoggingLevel_Info) GCDWebServerLogMessage(kGCDWebServerLoggingLevel_Info, __VA_ARGS__); } while (0)
  101. #define GWS_LOG_WARNING(...) do { if (GCDWebServerLogLevel <= kGCDWebServerLoggingLevel_Warning) GCDWebServerLogMessage(kGCDWebServerLoggingLevel_Warning, __VA_ARGS__); } while (0)
  102. #define GWS_LOG_ERROR(...) do { if (GCDWebServerLogLevel <= kGCDWebServerLoggingLevel_Error) GCDWebServerLogMessage(kGCDWebServerLoggingLevel_Error, __VA_ARGS__); } while (0)
  103. #endif
  104. /**
  105. * Consistency check macros used when building Debug only.
  106. */
  107. #if !defined(GWS_DCHECK) || !defined(GWS_DNOT_REACHED)
  108. #if DEBUG
  109. #define GWS_DCHECK(__CONDITION__) \
  110. do { \
  111. if (!(__CONDITION__)) { \
  112. abort(); \
  113. } \
  114. } while (0)
  115. #define GWS_DNOT_REACHED() abort()
  116. #else
  117. #define GWS_DCHECK(__CONDITION__)
  118. #define GWS_DNOT_REACHED()
  119. #endif
  120. #endif
  121. /**
  122. * GCDWebServer internal constants and APIs.
  123. */
  124. #define kGCDWebServerDefaultMimeType @"application/octet-stream"
  125. #define kGCDWebServerErrorDomain @"GCDWebServerErrorDomain"
  126. static inline BOOL GCDWebServerIsValidByteRange(NSRange range) {
  127. return ((range.location != NSUIntegerMax) || (range.length > 0));
  128. }
  129. static inline NSError* GCDWebServerMakePosixError(int code) {
  130. return [NSError errorWithDomain:NSPOSIXErrorDomain code:code userInfo:@{NSLocalizedDescriptionKey: [NSString stringWithUTF8String:strerror(code)]}];
  131. }
  132. extern void GCDWebServerInitializeFunctions();
  133. extern NSString* GCDWebServerNormalizeHeaderValue(NSString* value);
  134. extern NSString* GCDWebServerTruncateHeaderValue(NSString* value);
  135. extern NSString* GCDWebServerExtractHeaderValueParameter(NSString* header, NSString* attribute);
  136. extern NSStringEncoding GCDWebServerStringEncodingFromCharset(NSString* charset);
  137. extern BOOL GCDWebServerIsTextContentType(NSString* type);
  138. extern NSString* GCDWebServerDescribeData(NSData* data, NSString* contentType);
  139. extern NSString* GCDWebServerComputeMD5Digest(NSString* format, ...) NS_FORMAT_FUNCTION(1,2);
  140. extern NSString* GCDWebServerStringFromSockAddr(const struct sockaddr* addr, BOOL includeService);
  141. @interface GCDWebServerConnection ()
  142. - (id)initWithServer:(GCDWebServer*)server localAddress:(NSData*)localAddress remoteAddress:(NSData*)remoteAddress socket:(CFSocketNativeHandle)socket;
  143. @end
  144. @interface GCDWebServer ()
  145. @property(nonatomic, readonly) NSArray* handlers;
  146. @property(nonatomic, readonly) NSString* serverName;
  147. @property(nonatomic, readonly) NSString* authenticationRealm;
  148. @property(nonatomic, readonly) NSDictionary* authenticationBasicAccounts;
  149. @property(nonatomic, readonly) NSDictionary* authenticationDigestAccounts;
  150. @property(nonatomic, readonly) BOOL shouldAutomaticallyMapHEADToGET;
  151. @property(nonatomic, readonly) dispatch_queue_priority_t dispatchQueuePriority;
  152. - (void)willStartConnection:(GCDWebServerConnection*)connection;
  153. - (void)didEndConnection:(GCDWebServerConnection*)connection;
  154. @end
  155. @interface GCDWebServerHandler : NSObject
  156. @property(nonatomic, readonly) GCDWebServerMatchBlock matchBlock;
  157. @property(nonatomic, readonly) GCDWebServerAsyncProcessBlock asyncProcessBlock;
  158. @end
  159. @interface GCDWebServerRequest ()
  160. @property(nonatomic, readonly) BOOL usesChunkedTransferEncoding;
  161. @property(nonatomic, readwrite) NSData* localAddressData;
  162. @property(nonatomic, readwrite) NSData* remoteAddressData;
  163. - (void)prepareForWriting;
  164. - (BOOL)performOpen:(NSError**)error;
  165. - (BOOL)performWriteData:(NSData*)data error:(NSError**)error;
  166. - (BOOL)performClose:(NSError**)error;
  167. - (void)setAttribute:(id)attribute forKey:(NSString*)key;
  168. @end
  169. @interface GCDWebServerResponse ()
  170. @property(nonatomic, readonly) NSDictionary* additionalHeaders;
  171. @property(nonatomic, readonly) BOOL usesChunkedTransferEncoding;
  172. - (void)prepareForReading;
  173. - (BOOL)performOpen:(NSError**)error;
  174. - (void)performReadDataWithCompletion:(GCDWebServerBodyReaderCompletionBlock)block;
  175. - (void)performClose;
  176. @end