blob: d493f7d92357b011ca63987386ce2d332f5bde6b [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
8#include "GrCaps.h"
9#include "GrContextFactory.h"
joshualitt24dd6872016-02-25 08:37:54 -080010
11#include "Request.h"
joshualitt26cc3f52016-02-25 11:09:36 -080012#include "Response.h"
joshualitt24dd6872016-02-25 08:37:54 -080013
joshualitt7f6a1e02016-01-22 11:21:43 -080014#include "SkCommandLineFlags.h"
joshualittcdad12f2016-02-08 07:08:21 -080015
joshualitt3854f112016-02-25 11:28:18 -080016#include "microhttpd.h"
17
18#include "urlhandlers/UrlHandler.h"
joshualitt7f6a1e02016-01-22 11:21:43 -080019#include <sys/socket.h>
joshualitt7f6a1e02016-01-22 11:21:43 -080020
joshualitt26cc3f52016-02-25 11:09:36 -080021using namespace Response;
22
joshualitt7f6a1e02016-01-22 11:21:43 -080023// To get image decoders linked in we have to do the below magic
24#include "SkForceLinking.h"
25#include "SkImageDecoder.h"
26__SK_FORCE_IMAGE_DECODER_LINKING;
27
jcgregorio21ab1202016-01-28 06:24:19 -080028DEFINE_int32(port, 8888, "The port to listen on.");
joshualitt7f6a1e02016-01-22 11:21:43 -080029
joshualittccfdaa52016-01-27 07:40:29 -080030class UrlManager {
31public:
32 UrlManager() {
33 // Register handlers
joshualitt483b9012016-02-02 07:16:24 -080034 fHandlers.push_back(new RootHandler);
35 fHandlers.push_back(new PostHandler);
36 fHandlers.push_back(new ImgHandler);
ethannicholas0a0520a2016-02-12 12:06:53 -080037 fHandlers.push_back(new ClipAlphaHandler);
ethannicholas85fca852016-02-19 08:40:59 -080038 fHandlers.push_back(new EnableGPUHandler);
joshualitt29e5a892016-02-04 06:08:33 -080039 fHandlers.push_back(new CmdHandler);
joshualitt483b9012016-02-02 07:16:24 -080040 fHandlers.push_back(new InfoHandler);
joshualitt792345f2016-02-02 13:02:33 -080041 fHandlers.push_back(new DownloadHandler);
joshualittcdad12f2016-02-08 07:08:21 -080042 fHandlers.push_back(new DataHandler);
ethannicholas3ca1e1a2016-02-18 10:22:34 -080043 fHandlers.push_back(new BreakHandler);
joshualitt1e5884b2016-02-26 08:22:49 -080044 fHandlers.push_back(new BatchesHandler);
joshualitt483b9012016-02-02 07:16:24 -080045 }
46
47 ~UrlManager() {
48 for (int i = 0; i < fHandlers.count(); i++) { delete fHandlers[i]; }
joshualittccfdaa52016-01-27 07:40:29 -080049 }
50
51 // This is clearly not efficient for a large number of urls and handlers
52 int invoke(Request* request, MHD_Connection* connection, const char* url, const char* method,
53 const char* upload_data, size_t* upload_data_size) const {
54 for (int i = 0; i < fHandlers.count(); i++) {
joshualitt483b9012016-02-02 07:16:24 -080055 if (fHandlers[i]->canHandle(method, url)) {
joshualitt136f5172016-02-02 11:07:39 -080056 return fHandlers[i]->handle(request, connection, url, method, upload_data,
57 upload_data_size);
joshualittccfdaa52016-01-27 07:40:29 -080058 }
59 }
60 return MHD_NO;
61 }
62
63private:
joshualitt483b9012016-02-02 07:16:24 -080064 SkTArray<UrlHandler*> fHandlers;
joshualittccfdaa52016-01-27 07:40:29 -080065};
66
67const UrlManager kUrlManager;
68
joshualitt7f6a1e02016-01-22 11:21:43 -080069int answer_to_connection(void* cls, struct MHD_Connection* connection,
70 const char* url, const char* method, const char* version,
71 const char* upload_data, size_t* upload_data_size,
72 void** con_cls) {
joshualitt9a4e1882016-01-27 07:03:29 -080073 SkDebugf("New %s request for %s using version %s\n", method, url, version);
joshualitt7f6a1e02016-01-22 11:21:43 -080074
joshualitt9a4e1882016-01-27 07:03:29 -080075 Request* request = reinterpret_cast<Request*>(cls);
joshualitt483b9012016-02-02 07:16:24 -080076 int result = kUrlManager.invoke(request, connection, url, method, upload_data,
77 upload_data_size);
78 if (MHD_NO == result) {
joshualitt873d6242016-02-08 13:57:44 -080079 fprintf(stderr, "Invalid method and / or url: %s %s\n", method, url);
joshualitt483b9012016-02-02 07:16:24 -080080 }
81 return result;
joshualitt7f6a1e02016-01-22 11:21:43 -080082}
83
84int skiaserve_main() {
joshualittcdad12f2016-02-08 07:08:21 -080085 Request request(SkString("/data")); // This simple server has one request
ethannicholas85fca852016-02-19 08:40:59 -080086
87 // create surface
88 GrContextOptions grContextOpts;
89 request.fContextFactory.reset(new GrContextFactory(grContextOpts));
joshualitt4dcbe432016-02-25 10:50:28 -080090 request.fSurface.reset(request.createCPUSurface());
ethannicholas85fca852016-02-19 08:40:59 -080091
joshualitt7f6a1e02016-01-22 11:21:43 -080092 struct MHD_Daemon* daemon;
jcgregorio21ab1202016-01-28 06:24:19 -080093 // TODO Add option to bind this strictly to an address, e.g. localhost, for security.
94 daemon = MHD_start_daemon(MHD_USE_SELECT_INTERNALLY, FLAGS_port, nullptr, nullptr,
joshualitt9a4e1882016-01-27 07:03:29 -080095 &answer_to_connection, &request,
96 MHD_OPTION_END);
joshualitt7f6a1e02016-01-22 11:21:43 -080097 if (NULL == daemon) {
98 return 1;
99 }
100
101 getchar();
102 MHD_stop_daemon(daemon);
103 return 0;
104}
105
106#if !defined SK_BUILD_FOR_IOS
107int main(int argc, char** argv) {
108 SkCommandLineFlags::Parse(argc, argv);
109 return skiaserve_main();
110}
111#endif