main.m 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  1. /*
  2. Copyright (c) 2012-2019, 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 <libgen.h>
  26. #import "GCDWebServer.h"
  27. #import "GCDWebServerDataRequest.h"
  28. #import "GCDWebServerURLEncodedFormRequest.h"
  29. #import "GCDWebServerMultiPartFormRequest.h"
  30. #import "GCDWebServerDataResponse.h"
  31. #import "GCDWebServerStreamedResponse.h"
  32. #import "GCDWebDAVServer.h"
  33. #import "GCDWebUploader.h"
  34. #ifndef __GCDWEBSERVER_ENABLE_TESTING__
  35. #error __GCDWEBSERVER_ENABLE_TESTING__ must be defined
  36. #endif
  37. typedef enum {
  38. kMode_WebServer = 0,
  39. kMode_HTMLPage,
  40. kMode_HTMLForm,
  41. kMode_HTMLFileUpload,
  42. kMode_WebDAV,
  43. kMode_WebUploader,
  44. kMode_StreamingResponse,
  45. kMode_AsyncResponse
  46. } Mode;
  47. @interface Delegate : NSObject <GCDWebServerDelegate, GCDWebDAVServerDelegate, GCDWebUploaderDelegate>
  48. @end
  49. @implementation Delegate
  50. - (void)_logDelegateCall:(SEL)selector {
  51. fprintf(stdout, "<DELEGATE METHOD \"%s\" CALLED>\n", [NSStringFromSelector(selector) UTF8String]);
  52. }
  53. - (void)webServerDidStart:(GCDWebServer*)server {
  54. [self _logDelegateCall:_cmd];
  55. }
  56. - (void)webServerDidCompleteBonjourRegistration:(GCDWebServer*)server {
  57. [self _logDelegateCall:_cmd];
  58. }
  59. - (void)webServerDidUpdateNATPortMapping:(GCDWebServer*)server {
  60. [self _logDelegateCall:_cmd];
  61. }
  62. - (void)webServerDidConnect:(GCDWebServer*)server {
  63. [self _logDelegateCall:_cmd];
  64. }
  65. - (void)webServerDidDisconnect:(GCDWebServer*)server {
  66. [self _logDelegateCall:_cmd];
  67. }
  68. - (void)webServerDidStop:(GCDWebServer*)server {
  69. [self _logDelegateCall:_cmd];
  70. }
  71. - (void)davServer:(GCDWebDAVServer*)server didDownloadFileAtPath:(NSString*)path {
  72. [self _logDelegateCall:_cmd];
  73. }
  74. - (void)davServer:(GCDWebDAVServer*)server didUploadFileAtPath:(NSString*)path {
  75. [self _logDelegateCall:_cmd];
  76. }
  77. - (void)davServer:(GCDWebDAVServer*)server didMoveItemFromPath:(NSString*)fromPath toPath:(NSString*)toPath {
  78. [self _logDelegateCall:_cmd];
  79. }
  80. - (void)davServer:(GCDWebDAVServer*)server didCopyItemFromPath:(NSString*)fromPath toPath:(NSString*)toPath {
  81. [self _logDelegateCall:_cmd];
  82. }
  83. - (void)davServer:(GCDWebDAVServer*)server didDeleteItemAtPath:(NSString*)path {
  84. [self _logDelegateCall:_cmd];
  85. }
  86. - (void)davServer:(GCDWebDAVServer*)server didCreateDirectoryAtPath:(NSString*)path {
  87. [self _logDelegateCall:_cmd];
  88. }
  89. - (void)webUploader:(GCDWebUploader*)uploader didDownloadFileAtPath:(NSString*)path {
  90. [self _logDelegateCall:_cmd];
  91. }
  92. - (void)webUploader:(GCDWebUploader*)uploader didUploadFileAtPath:(NSString*)path {
  93. [self _logDelegateCall:_cmd];
  94. }
  95. - (void)webUploader:(GCDWebUploader*)uploader didMoveItemFromPath:(NSString*)fromPath toPath:(NSString*)toPath {
  96. [self _logDelegateCall:_cmd];
  97. }
  98. - (void)webUploader:(GCDWebUploader*)uploader didDeleteItemAtPath:(NSString*)path {
  99. [self _logDelegateCall:_cmd];
  100. }
  101. - (void)webUploader:(GCDWebUploader*)uploader didCreateDirectoryAtPath:(NSString*)path {
  102. [self _logDelegateCall:_cmd];
  103. }
  104. @end
  105. int main(int argc, const char* argv[]) {
  106. int result = -1;
  107. @autoreleasepool {
  108. Mode mode = kMode_WebServer;
  109. BOOL recording = NO;
  110. NSString* rootDirectory = NSHomeDirectory();
  111. NSString* testDirectory = nil;
  112. NSString* authenticationMethod = nil;
  113. NSString* authenticationRealm = nil;
  114. NSString* authenticationUser = nil;
  115. NSString* authenticationPassword = nil;
  116. BOOL bindToLocalhost = NO;
  117. BOOL requestNATPortMapping = NO;
  118. if (argc == 1) {
  119. fprintf(stdout, "Usage: %s [-mode webServer | htmlPage | htmlForm | htmlFileUpload | webDAV | webUploader | streamingResponse | asyncResponse] [-record] [-root directory] [-tests directory] [-authenticationMethod Basic | Digest] [-authenticationRealm realm] [-authenticationUser user] [-authenticationPassword password] [--localhost]\n\n", basename((char*)argv[0]));
  120. } else {
  121. for (int i = 1; i < argc; ++i) {
  122. if (argv[i][0] != '-') {
  123. continue;
  124. }
  125. if (!strcmp(argv[i], "-mode") && (i + 1 < argc)) {
  126. ++i;
  127. if (!strcmp(argv[i], "webServer")) {
  128. mode = kMode_WebServer;
  129. } else if (!strcmp(argv[i], "htmlPage")) {
  130. mode = kMode_HTMLPage;
  131. } else if (!strcmp(argv[i], "htmlForm")) {
  132. mode = kMode_HTMLForm;
  133. } else if (!strcmp(argv[i], "htmlFileUpload")) {
  134. mode = kMode_HTMLFileUpload;
  135. } else if (!strcmp(argv[i], "webDAV")) {
  136. mode = kMode_WebDAV;
  137. } else if (!strcmp(argv[i], "webUploader")) {
  138. mode = kMode_WebUploader;
  139. } else if (!strcmp(argv[i], "streamingResponse")) {
  140. mode = kMode_StreamingResponse;
  141. } else if (!strcmp(argv[i], "asyncResponse")) {
  142. mode = kMode_AsyncResponse;
  143. }
  144. } else if (!strcmp(argv[i], "-record")) {
  145. recording = YES;
  146. } else if (!strcmp(argv[i], "-root") && (i + 1 < argc)) {
  147. ++i;
  148. rootDirectory = [[NSFileManager defaultManager] stringWithFileSystemRepresentation:argv[i] length:strlen(argv[i])];
  149. } else if (!strcmp(argv[i], "-tests") && (i + 1 < argc)) {
  150. ++i;
  151. testDirectory = [[NSFileManager defaultManager] stringWithFileSystemRepresentation:argv[i] length:strlen(argv[i])];
  152. } else if (!strcmp(argv[i], "-authenticationMethod") && (i + 1 < argc)) {
  153. ++i;
  154. authenticationMethod = [NSString stringWithUTF8String:argv[i]];
  155. } else if (!strcmp(argv[i], "-authenticationRealm") && (i + 1 < argc)) {
  156. ++i;
  157. authenticationRealm = [NSString stringWithUTF8String:argv[i]];
  158. } else if (!strcmp(argv[i], "-authenticationUser") && (i + 1 < argc)) {
  159. ++i;
  160. authenticationUser = [NSString stringWithUTF8String:argv[i]];
  161. } else if (!strcmp(argv[i], "-authenticationPassword") && (i + 1 < argc)) {
  162. ++i;
  163. authenticationPassword = [NSString stringWithUTF8String:argv[i]];
  164. } else if (!strcmp(argv[i], "--localhost")) {
  165. bindToLocalhost = YES;
  166. } else if (!strcmp(argv[i], "--nat")) {
  167. requestNATPortMapping = YES;
  168. }
  169. }
  170. }
  171. GCDWebServer* webServer = nil;
  172. switch (mode) {
  173. // Simply serve contents of home directory
  174. case kMode_WebServer: {
  175. fprintf(stdout, "Running in Web Server mode from \"%s\"\n", [rootDirectory UTF8String]);
  176. webServer = [[GCDWebServer alloc] init];
  177. [webServer addGETHandlerForBasePath:@"/" directoryPath:rootDirectory indexFilename:nil cacheAge:0 allowRangeRequests:YES];
  178. break;
  179. }
  180. // Renders a HTML page
  181. case kMode_HTMLPage: {
  182. fprintf(stdout, "Running in HTML Page mode\n");
  183. webServer = [[GCDWebServer alloc] init];
  184. [webServer addDefaultHandlerForMethod:@"GET"
  185. requestClass:[GCDWebServerRequest class]
  186. processBlock:^GCDWebServerResponse*(GCDWebServerRequest* request) {
  187. return [GCDWebServerDataResponse responseWithHTML:@"<html><body><p>Hello World</p></body></html>"];
  188. }];
  189. break;
  190. }
  191. // Implements an HTML form
  192. case kMode_HTMLForm: {
  193. fprintf(stdout, "Running in HTML Form mode\n");
  194. webServer = [[GCDWebServer alloc] init];
  195. [webServer addHandlerForMethod:@"GET"
  196. path:@"/"
  197. requestClass:[GCDWebServerRequest class]
  198. processBlock:^GCDWebServerResponse*(GCDWebServerRequest* request) {
  199. NSString* html = @" \
  200. <html><body> \
  201. <form name=\"input\" action=\"/\" method=\"post\" enctype=\"application/x-www-form-urlencoded\"> \
  202. Value: <input type=\"text\" name=\"value\"> \
  203. <input type=\"submit\" value=\"Submit\"> \
  204. </form> \
  205. </body></html> \
  206. ";
  207. return [GCDWebServerDataResponse responseWithHTML:html];
  208. }];
  209. [webServer addHandlerForMethod:@"POST"
  210. path:@"/"
  211. requestClass:[GCDWebServerURLEncodedFormRequest class]
  212. processBlock:^GCDWebServerResponse*(GCDWebServerRequest* request) {
  213. NSString* value = [[(GCDWebServerURLEncodedFormRequest*)request arguments] objectForKey:@"value"];
  214. NSString* html = [NSString stringWithFormat:@"<html><body><p>%@</p></body></html>", value];
  215. return [GCDWebServerDataResponse responseWithHTML:html];
  216. }];
  217. break;
  218. }
  219. // Implements HTML file upload
  220. case kMode_HTMLFileUpload: {
  221. fprintf(stdout, "Running in HTML File Upload mode\n");
  222. webServer = [[GCDWebServer alloc] init];
  223. NSString* formHTML = @" \
  224. <form name=\"input\" action=\"/\" method=\"post\" enctype=\"multipart/form-data\"> \
  225. <input type=\"hidden\" name=\"secret\" value=\"42\"> \
  226. <input type=\"file\" name=\"files\" multiple><br/> \
  227. <input type=\"submit\" value=\"Submit\"> \
  228. </form> \
  229. ";
  230. [webServer addHandlerForMethod:@"GET"
  231. path:@"/"
  232. requestClass:[GCDWebServerRequest class]
  233. processBlock:^GCDWebServerResponse*(GCDWebServerRequest* request) {
  234. NSString* html = [NSString stringWithFormat:@"<html><body>%@</body></html>", formHTML];
  235. return [GCDWebServerDataResponse responseWithHTML:html];
  236. }];
  237. [webServer addHandlerForMethod:@"POST"
  238. path:@"/"
  239. requestClass:[GCDWebServerMultiPartFormRequest class]
  240. processBlock:^GCDWebServerResponse*(GCDWebServerRequest* request) {
  241. NSMutableString* string = [NSMutableString string];
  242. for (GCDWebServerMultiPartArgument* argument in [(GCDWebServerMultiPartFormRequest*)request arguments]) {
  243. [string appendFormat:@"%@ = %@<br>", argument.controlName, argument.string];
  244. }
  245. for (GCDWebServerMultiPartFile* file in [(GCDWebServerMultiPartFormRequest*)request files]) {
  246. NSDictionary* attributes = [[NSFileManager defaultManager] attributesOfItemAtPath:file.temporaryPath error:NULL];
  247. [string appendFormat:@"%@ = &quot;%@&quot; (%@ | %llu %@)<br>", file.controlName, file.fileName, file.mimeType,
  248. attributes.fileSize >= 1000 ? attributes.fileSize / 1000 : attributes.fileSize,
  249. attributes.fileSize >= 1000 ? @"KB" : @"Bytes"];
  250. };
  251. NSString* html = [NSString stringWithFormat:@"<html><body><p>%@</p><hr>%@</body></html>", string, formHTML];
  252. return [GCDWebServerDataResponse responseWithHTML:html];
  253. }];
  254. break;
  255. }
  256. // Serve home directory through WebDAV
  257. case kMode_WebDAV: {
  258. fprintf(stdout, "Running in WebDAV mode from \"%s\"\n", [rootDirectory UTF8String]);
  259. webServer = [[GCDWebDAVServer alloc] initWithUploadDirectory:rootDirectory];
  260. break;
  261. }
  262. // Serve home directory through web uploader
  263. case kMode_WebUploader: {
  264. fprintf(stdout, "Running in Web Uploader mode from \"%s\"\n", [rootDirectory UTF8String]);
  265. webServer = [[GCDWebUploader alloc] initWithUploadDirectory:rootDirectory];
  266. break;
  267. }
  268. // Test streaming responses
  269. case kMode_StreamingResponse: {
  270. fprintf(stdout, "Running in Streaming Response mode\n");
  271. webServer = [[GCDWebServer alloc] init];
  272. [webServer addHandlerForMethod:@"GET"
  273. path:@"/sync"
  274. requestClass:[GCDWebServerRequest class]
  275. processBlock:^GCDWebServerResponse*(GCDWebServerRequest* request) {
  276. __block int countDown = 10;
  277. return [GCDWebServerStreamedResponse responseWithContentType:@"text/plain"
  278. streamBlock:^NSData*(NSError** error) {
  279. usleep(100 * 1000);
  280. if (countDown) {
  281. return [[NSString stringWithFormat:@"%i\n", countDown--] dataUsingEncoding:NSUTF8StringEncoding];
  282. } else {
  283. return [NSData data];
  284. }
  285. }];
  286. }];
  287. [webServer addHandlerForMethod:@"GET"
  288. path:@"/async"
  289. requestClass:[GCDWebServerRequest class]
  290. processBlock:^GCDWebServerResponse*(GCDWebServerRequest* request) {
  291. __block int countDown = 10;
  292. return [GCDWebServerStreamedResponse responseWithContentType:@"text/plain"
  293. asyncStreamBlock:^(GCDWebServerBodyReaderCompletionBlock completionBlock) {
  294. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  295. NSData* data = countDown ? [[NSString stringWithFormat:@"%i\n", countDown--] dataUsingEncoding:NSUTF8StringEncoding] : [NSData data];
  296. completionBlock(data, nil);
  297. });
  298. }];
  299. }];
  300. break;
  301. }
  302. // Test async responses
  303. case kMode_AsyncResponse: {
  304. fprintf(stdout, "Running in Async Response mode\n");
  305. webServer = [[GCDWebServer alloc] init];
  306. [webServer addHandlerForMethod:@"GET"
  307. path:@"/async"
  308. requestClass:[GCDWebServerRequest class]
  309. asyncProcessBlock:^(GCDWebServerRequest* request, GCDWebServerCompletionBlock completionBlock) {
  310. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1 * NSEC_PER_SEC)), dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  311. GCDWebServerDataResponse* response = [GCDWebServerDataResponse responseWithData:(NSData*)[@"Hello World!" dataUsingEncoding:NSUTF8StringEncoding] contentType:@"text/plain"];
  312. completionBlock(response);
  313. });
  314. }];
  315. [webServer addHandlerForMethod:@"GET"
  316. path:@"/async2"
  317. requestClass:[GCDWebServerRequest class]
  318. asyncProcessBlock:^(GCDWebServerRequest* request, GCDWebServerCompletionBlock handlerCompletionBlock) {
  319. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1 * NSEC_PER_SEC)), dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  320. __block int countDown = 10;
  321. GCDWebServerStreamedResponse* response = [GCDWebServerStreamedResponse responseWithContentType:@"text/plain"
  322. asyncStreamBlock:^(GCDWebServerBodyReaderCompletionBlock readerCompletionBlock) {
  323. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  324. NSData* data = countDown ? [[NSString stringWithFormat:@"%i\n", countDown--] dataUsingEncoding:NSUTF8StringEncoding] : [NSData data];
  325. readerCompletionBlock(data, nil);
  326. });
  327. }];
  328. handlerCompletionBlock(response);
  329. });
  330. }];
  331. break;
  332. }
  333. }
  334. if (webServer) {
  335. Delegate* delegate = [[Delegate alloc] init];
  336. if (testDirectory) {
  337. #if DEBUG
  338. webServer.delegate = delegate;
  339. #endif
  340. fprintf(stdout, "<RUNNING TESTS FROM \"%s\">\n\n", [testDirectory UTF8String]);
  341. result = (int)[webServer runTestsWithOptions:@{GCDWebServerOption_Port : @8080} inDirectory:testDirectory];
  342. } else {
  343. webServer.delegate = delegate;
  344. if (recording) {
  345. fprintf(stdout, "<RECORDING ENABLED>\n");
  346. webServer.recordingEnabled = YES;
  347. }
  348. fprintf(stdout, "\n");
  349. NSMutableDictionary* options = [NSMutableDictionary dictionary];
  350. [options setObject:@8080 forKey:GCDWebServerOption_Port];
  351. [options setObject:@(requestNATPortMapping) forKey:GCDWebServerOption_RequestNATPortMapping];
  352. [options setObject:@(bindToLocalhost) forKey:GCDWebServerOption_BindToLocalhost];
  353. [options setObject:@"" forKey:GCDWebServerOption_BonjourName];
  354. if (authenticationUser && authenticationPassword) {
  355. [options setValue:authenticationRealm forKey:GCDWebServerOption_AuthenticationRealm];
  356. [options setObject:@{authenticationUser : authenticationPassword} forKey:GCDWebServerOption_AuthenticationAccounts];
  357. if ([authenticationMethod isEqualToString:@"Basic"]) {
  358. [options setObject:GCDWebServerAuthenticationMethod_Basic forKey:GCDWebServerOption_AuthenticationMethod];
  359. } else if ([authenticationMethod isEqualToString:@"Digest"]) {
  360. [options setObject:GCDWebServerAuthenticationMethod_DigestAccess forKey:GCDWebServerOption_AuthenticationMethod];
  361. }
  362. }
  363. if ([webServer runWithOptions:options error:NULL]) {
  364. result = 0;
  365. }
  366. }
  367. webServer.delegate = nil;
  368. }
  369. }
  370. return result;
  371. }