huangcm
2025-02-24 69ed55dec4b2116a19e4cca4393cbc014fce5fb2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
/*
 * Copyright 2014 Google Inc. All rights reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
 
#include "server.h"
 
#include <ctime>
#include <iostream>
#include <thread>
 
using namespace std;
using namespace fruit;
 
class ServerImpl : public Server {
private:
  std::vector<std::thread> threads;
 
public:
  INJECT(ServerImpl()) {}
 
  ~ServerImpl() {
    for (std::thread& t : threads) {
      t.join();
    }
  }
 
  void run(Component<Required<Request, ServerContext>, RequestDispatcher> (*getRequestDispatcherComponent)()) override {
    ServerContext serverContext;
    serverContext.startupTime = getTime();
 
    const NormalizedComponent<Required<Request>, RequestDispatcher> requestDispatcherNormalizedComponent(
        getRequestDispatcherComponentWithContext, getRequestDispatcherComponent, &serverContext);
 
    cerr << "Server started." << endl;
 
    while (1) {
      cerr << endl;
      cerr << "Enter the request (absolute path starting with \"/foo/\" or \"/bar/\"), or an empty line to exit."
           << endl;
      Request request;
      getline(cin, request.path);
      cerr << "Server received request: " + request.path << endl;
      if (request.path.empty()) {
        cerr << "Server received empty line, shutting down." << endl;
        break;
      }
 
      // In production code we would use a thread pool.
      // Here we spawn a new thread each time to keep it simple.
      threads.push_back(std::thread(worker_thread_main, std::ref(requestDispatcherNormalizedComponent), request));
    }
  }
 
private:
  static void worker_thread_main(
      const NormalizedComponent<Required<Request>, RequestDispatcher>& requestDispatcherNormalizedComponent,
      Request request) {
    Injector<RequestDispatcher> injector(requestDispatcherNormalizedComponent, getRequestComponent, &request);
 
    RequestDispatcher* requestDispatcher(injector);
    requestDispatcher->handleRequest();
  }
 
  static string getTime() {
    time_t now = time(nullptr);
    tm* localTime = localtime(&now);
    string result = asctime(localTime);
    if (result.size() != 0 && result.back() == '\n') {
      result.pop_back();
    }
    return result;
  }
 
  static Component<Request> getRequestComponent(Request* request) {
    return createComponent().bindInstance(*request);
  }
 
  static Component<Required<Request>, RequestDispatcher> getRequestDispatcherComponentWithContext(
      Component<Required<Request, ServerContext>, RequestDispatcher> (*getRequestDispatcherComponent)(),
      ServerContext* serverContext) {
    return createComponent().install(getRequestDispatcherComponent).bindInstance(*serverContext);
  }
};
 
fruit::Component<Server> getServerComponent() {
  return fruit::createComponent().bind<Server, ServerImpl>();
}