blob: 29dd1b4996b2e5ab91441dfa39addc600abe28f0 [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"
mtklein05db63b2016-04-19 12:42:24 -070012#include "SkGraphics.h"
joshualittcdad12f2016-02-08 07:08:21 -080013
joshualitt3854f112016-02-25 11:28:18 -080014#include "microhttpd.h"
15
16#include "urlhandlers/UrlHandler.h"
joshualittb82f0882016-03-08 13:24:33 -080017
18#include <errno.h>
brianosman82996b82016-04-20 10:52:54 -070019
20#if !defined _WIN32
joshualitt7f6a1e02016-01-22 11:21:43 -080021#include <sys/socket.h>
jcgregorio855bd442016-03-02 09:03:01 -080022#include <arpa/inet.h>
brianosman82996b82016-04-20 10:52:54 -070023#endif
joshualitt7f6a1e02016-01-22 11:21:43 -080024
joshualitt26cc3f52016-02-25 11:09:36 -080025using namespace Response;
26
jcgregorio21ab1202016-01-28 06:24:19 -080027DEFINE_int32(port, 8888, "The port to listen on.");
joshualittb82f0882016-03-08 13:24:33 -080028DEFINE_string(address, "127.0.0.1", "The address to bind to.");
jcgregorio34604042016-06-03 07:26:22 -070029DEFINE_bool(hosted, false, "Running in hosted mode on debugger.skia.org.");
joshualitt7f6a1e02016-01-22 11:21:43 -080030
joshualittccfdaa52016-01-27 07:40:29 -080031class UrlManager {
32public:
33 UrlManager() {
34 // Register handlers
joshualitt483b9012016-02-02 07:16:24 -080035 fHandlers.push_back(new RootHandler);
36 fHandlers.push_back(new PostHandler);
37 fHandlers.push_back(new ImgHandler);
ethannicholas0a0520a2016-02-12 12:06:53 -080038 fHandlers.push_back(new ClipAlphaHandler);
ethannicholas85fca852016-02-19 08:40:59 -080039 fHandlers.push_back(new EnableGPUHandler);
joshualitt29e5a892016-02-04 06:08:33 -080040 fHandlers.push_back(new CmdHandler);
joshualitt483b9012016-02-02 07:16:24 -080041 fHandlers.push_back(new InfoHandler);
joshualitt792345f2016-02-02 13:02:33 -080042 fHandlers.push_back(new DownloadHandler);
joshualittcdad12f2016-02-08 07:08:21 -080043 fHandlers.push_back(new DataHandler);
ethannicholas3ca1e1a2016-02-18 10:22:34 -080044 fHandlers.push_back(new BreakHandler);
Brian Salomon144a5c52016-12-20 16:48:59 -050045 fHandlers.push_back(new OpsHandler);
46 fHandlers.push_back(new OpBoundsHandler);
brianosman78312952016-04-19 10:16:53 -070047 fHandlers.push_back(new ColorModeHandler);
jcgregorioce8ea4c2016-05-31 12:51:31 -070048 fHandlers.push_back(new QuitHandler);
joshualitt483b9012016-02-02 07:16:24 -080049 }
50
51 ~UrlManager() {
52 for (int i = 0; i < fHandlers.count(); i++) { delete fHandlers[i]; }
joshualittccfdaa52016-01-27 07:40:29 -080053 }
54
55 // This is clearly not efficient for a large number of urls and handlers
56 int invoke(Request* request, MHD_Connection* connection, const char* url, const char* method,
57 const char* upload_data, size_t* upload_data_size) const {
58 for (int i = 0; i < fHandlers.count(); i++) {
joshualitt483b9012016-02-02 07:16:24 -080059 if (fHandlers[i]->canHandle(method, url)) {
joshualitt136f5172016-02-02 11:07:39 -080060 return fHandlers[i]->handle(request, connection, url, method, upload_data,
61 upload_data_size);
joshualittccfdaa52016-01-27 07:40:29 -080062 }
63 }
64 return MHD_NO;
65 }
66
67private:
joshualitt483b9012016-02-02 07:16:24 -080068 SkTArray<UrlHandler*> fHandlers;
joshualittccfdaa52016-01-27 07:40:29 -080069};
70
71const UrlManager kUrlManager;
72
joshualitt7f6a1e02016-01-22 11:21:43 -080073int answer_to_connection(void* cls, struct MHD_Connection* connection,
74 const char* url, const char* method, const char* version,
75 const char* upload_data, size_t* upload_data_size,
76 void** con_cls) {
joshualitt9a4e1882016-01-27 07:03:29 -080077 SkDebugf("New %s request for %s using version %s\n", method, url, version);
joshualitt7f6a1e02016-01-22 11:21:43 -080078
joshualitt9a4e1882016-01-27 07:03:29 -080079 Request* request = reinterpret_cast<Request*>(cls);
joshualitt483b9012016-02-02 07:16:24 -080080 int result = kUrlManager.invoke(request, connection, url, method, upload_data,
81 upload_data_size);
82 if (MHD_NO == result) {
joshualitt873d6242016-02-08 13:57:44 -080083 fprintf(stderr, "Invalid method and / or url: %s %s\n", method, url);
joshualitt483b9012016-02-02 07:16:24 -080084 }
85 return result;
joshualitt7f6a1e02016-01-22 11:21:43 -080086}
87
88int skiaserve_main() {
mtklein05db63b2016-04-19 12:42:24 -070089 SkGraphics::Init();
joshualittcdad12f2016-02-08 07:08:21 -080090 Request request(SkString("/data")); // This simple server has one request
ethannicholas85fca852016-02-19 08:40:59 -080091
jcgregorio855bd442016-03-02 09:03:01 -080092 struct sockaddr_in address;
93 address.sin_family = AF_INET;
94 address.sin_port = htons(FLAGS_port);
joshualittb82f0882016-03-08 13:24:33 -080095 int result = inet_pton(AF_INET, FLAGS_address[0], &address.sin_addr);
96 if (result != 1) {
halcanary9d524f22016-03-29 09:03:52 -070097 printf("inet_pton for %s:%d failed with return %d %s\n",
joshualittb82f0882016-03-08 13:24:33 -080098 FLAGS_address[0], FLAGS_port, result, strerror(errno));
99 return 1;
100 }
jcgregorio855bd442016-03-02 09:03:01 -0800101
102 printf("Visit http://%s:%d in your browser.\n", FLAGS_address[0], FLAGS_port);
103
joshualitt7f6a1e02016-01-22 11:21:43 -0800104 struct MHD_Daemon* daemon;
joshualitte43f7e62016-03-04 10:45:05 -0800105 daemon = MHD_start_daemon(MHD_USE_SELECT_INTERNALLY
106#ifdef SK_DEBUG
107 | MHD_USE_DEBUG
halcanary9d524f22016-03-29 09:03:52 -0700108#endif
joshualitte43f7e62016-03-04 10:45:05 -0800109 , FLAGS_port, nullptr, nullptr,
joshualitt9a4e1882016-01-27 07:03:29 -0800110 &answer_to_connection, &request,
jcgregorio855bd442016-03-02 09:03:01 -0800111 MHD_OPTION_SOCK_ADDR, &address,
joshualitt9a4e1882016-01-27 07:03:29 -0800112 MHD_OPTION_END);
joshualitt7f6a1e02016-01-22 11:21:43 -0800113 if (NULL == daemon) {
joshualitte43f7e62016-03-04 10:45:05 -0800114 SkDebugf("Could not initialize daemon\n");
joshualitt7f6a1e02016-01-22 11:21:43 -0800115 return 1;
116 }
117
jcgregorio34604042016-06-03 07:26:22 -0700118 if (FLAGS_hosted) {
119 while (1) {
120 SkDebugf("loop\n");
121 #if defined(SK_BUILD_FOR_WIN)
122 Sleep(60 * 1000);
123 #else
124 sleep(60);
125 #endif
126 }
127 } else {
128 getchar();
129 }
joshualitt7f6a1e02016-01-22 11:21:43 -0800130 MHD_stop_daemon(daemon);
131 return 0;
132}
133
134#if !defined SK_BUILD_FOR_IOS
135int main(int argc, char** argv) {
136 SkCommandLineFlags::Parse(argc, argv);
137 return skiaserve_main();
138}
139#endif