blob: 889eaafebd2d9c86ff2a4d37d6a366b2c1e5f216 [file] [log] [blame]
joshualitt7f6a1e02016-01-22 11:21:43 -08001/*
2 * Copyright 2016 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
joshualitt24dd6872016-02-25 08:37:54 -08008#include "Request.h"
joshualitt26cc3f52016-02-25 11:09:36 -08009#include "Response.h"
joshualitt24dd6872016-02-25 08:37:54 -080010
joshualitt7f6a1e02016-01-22 11:21:43 -080011#include "SkCommandLineFlags.h"
joshualittcdad12f2016-02-08 07:08:21 -080012
joshualitt3854f112016-02-25 11:28:18 -080013#include "microhttpd.h"
14
15#include "urlhandlers/UrlHandler.h"
joshualitt7f6a1e02016-01-22 11:21:43 -080016#include <sys/socket.h>
jcgregorio855bd442016-03-02 09:03:01 -080017#include <arpa/inet.h>
joshualitt7f6a1e02016-01-22 11:21:43 -080018
joshualitt26cc3f52016-02-25 11:09:36 -080019using namespace Response;
20
jcgregorio21ab1202016-01-28 06:24:19 -080021DEFINE_int32(port, 8888, "The port to listen on.");
jcgregorio855bd442016-03-02 09:03:01 -080022DEFINE_string(address, "localhost", "The address to bind to.");
joshualitt7f6a1e02016-01-22 11:21:43 -080023
joshualittccfdaa52016-01-27 07:40:29 -080024class UrlManager {
25public:
26 UrlManager() {
27 // Register handlers
joshualitt483b9012016-02-02 07:16:24 -080028 fHandlers.push_back(new RootHandler);
29 fHandlers.push_back(new PostHandler);
30 fHandlers.push_back(new ImgHandler);
ethannicholas0a0520a2016-02-12 12:06:53 -080031 fHandlers.push_back(new ClipAlphaHandler);
ethannicholas85fca852016-02-19 08:40:59 -080032 fHandlers.push_back(new EnableGPUHandler);
joshualitt29e5a892016-02-04 06:08:33 -080033 fHandlers.push_back(new CmdHandler);
joshualitt483b9012016-02-02 07:16:24 -080034 fHandlers.push_back(new InfoHandler);
joshualitt792345f2016-02-02 13:02:33 -080035 fHandlers.push_back(new DownloadHandler);
joshualittcdad12f2016-02-08 07:08:21 -080036 fHandlers.push_back(new DataHandler);
ethannicholas3ca1e1a2016-02-18 10:22:34 -080037 fHandlers.push_back(new BreakHandler);
joshualitt1e5884b2016-02-26 08:22:49 -080038 fHandlers.push_back(new BatchesHandler);
joshualitt10d8fc22016-02-29 11:15:06 -080039 fHandlers.push_back(new BatchBoundsHandler);
joshualitt483b9012016-02-02 07:16:24 -080040 }
41
42 ~UrlManager() {
43 for (int i = 0; i < fHandlers.count(); i++) { delete fHandlers[i]; }
joshualittccfdaa52016-01-27 07:40:29 -080044 }
45
46 // This is clearly not efficient for a large number of urls and handlers
47 int invoke(Request* request, MHD_Connection* connection, const char* url, const char* method,
48 const char* upload_data, size_t* upload_data_size) const {
49 for (int i = 0; i < fHandlers.count(); i++) {
joshualitt483b9012016-02-02 07:16:24 -080050 if (fHandlers[i]->canHandle(method, url)) {
joshualitt136f5172016-02-02 11:07:39 -080051 return fHandlers[i]->handle(request, connection, url, method, upload_data,
52 upload_data_size);
joshualittccfdaa52016-01-27 07:40:29 -080053 }
54 }
55 return MHD_NO;
56 }
57
58private:
joshualitt483b9012016-02-02 07:16:24 -080059 SkTArray<UrlHandler*> fHandlers;
joshualittccfdaa52016-01-27 07:40:29 -080060};
61
62const UrlManager kUrlManager;
63
joshualitt7f6a1e02016-01-22 11:21:43 -080064int answer_to_connection(void* cls, struct MHD_Connection* connection,
65 const char* url, const char* method, const char* version,
66 const char* upload_data, size_t* upload_data_size,
67 void** con_cls) {
joshualitt9a4e1882016-01-27 07:03:29 -080068 SkDebugf("New %s request for %s using version %s\n", method, url, version);
joshualitt7f6a1e02016-01-22 11:21:43 -080069
joshualitt9a4e1882016-01-27 07:03:29 -080070 Request* request = reinterpret_cast<Request*>(cls);
joshualitt483b9012016-02-02 07:16:24 -080071 int result = kUrlManager.invoke(request, connection, url, method, upload_data,
72 upload_data_size);
73 if (MHD_NO == result) {
joshualitt873d6242016-02-08 13:57:44 -080074 fprintf(stderr, "Invalid method and / or url: %s %s\n", method, url);
joshualitt483b9012016-02-02 07:16:24 -080075 }
76 return result;
joshualitt7f6a1e02016-01-22 11:21:43 -080077}
78
79int skiaserve_main() {
joshualittcdad12f2016-02-08 07:08:21 -080080 Request request(SkString("/data")); // This simple server has one request
ethannicholas85fca852016-02-19 08:40:59 -080081
jcgregorio855bd442016-03-02 09:03:01 -080082 struct sockaddr_in address;
83 address.sin_family = AF_INET;
84 address.sin_port = htons(FLAGS_port);
85 inet_pton(AF_INET, FLAGS_address[0], &address.sin_addr);
86
87 printf("Visit http://%s:%d in your browser.\n", FLAGS_address[0], FLAGS_port);
88
joshualitt7f6a1e02016-01-22 11:21:43 -080089 struct MHD_Daemon* daemon;
joshualitte43f7e62016-03-04 10:45:05 -080090 daemon = MHD_start_daemon(MHD_USE_SELECT_INTERNALLY
91#ifdef SK_DEBUG
92 | MHD_USE_DEBUG
93#endif
94 , FLAGS_port, nullptr, nullptr,
joshualitt9a4e1882016-01-27 07:03:29 -080095 &answer_to_connection, &request,
jcgregorio855bd442016-03-02 09:03:01 -080096 MHD_OPTION_SOCK_ADDR, &address,
joshualitt9a4e1882016-01-27 07:03:29 -080097 MHD_OPTION_END);
joshualitt7f6a1e02016-01-22 11:21:43 -080098 if (NULL == daemon) {
joshualitte43f7e62016-03-04 10:45:05 -080099 SkDebugf("Could not initialize daemon\n");
joshualitt7f6a1e02016-01-22 11:21:43 -0800100 return 1;
101 }
102
103 getchar();
104 MHD_stop_daemon(daemon);
105 return 0;
106}
107
108#if !defined SK_BUILD_FOR_IOS
109int main(int argc, char** argv) {
110 SkCommandLineFlags::Parse(argc, argv);
111 return skiaserve_main();
112}
113#endif