summaryrefslogtreecommitdiff
path: root/test/fixtures/fixture_request.cpp
blob: b106dfe306bf025b9dcfd7daf8be79f817433f6f (plain)
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
100
101
102
103
104
105
106
#include <llmr/platform/platform.hpp>
#include <llmr/platform/request.hpp>
#include <llmr/util/uv.hpp>
#include <llmr/platform/log.hpp>

const std::string base_directory = []{
    std::string fn = __FILE__;
    fn.erase(fn.find_last_of("/"));
    return fn;
}();


namespace llmr {

std::shared_ptr<platform::Request>
platform::request_http(const std::string &url,
                       std::function<void(Response *)> callback,
                       std::shared_ptr<uv::loop> loop) {
    uv_loop_t *l = nullptr;
    if (loop) {
        l = **loop;
    } else {
        l = uv_default_loop();
    }

    std::string clean_url = base_directory + "/" + url;
    auto pos = clean_url.find("://");
    if (pos != std::string::npos) {
        clean_url.replace(pos, 3, "/");
    }

    std::replace(clean_url.begin(), clean_url.end(), '+', ' ');

    std::shared_ptr<Request> req = std::make_shared<Request>(url, callback, loop);

    int err;

    uv_fs_t open_req;
    err = uv_fs_open(l, &open_req, clean_url.c_str(), O_RDONLY, S_IRUSR, nullptr);
    uv_fs_req_cleanup(&open_req);
    if (err < 0) {
        req->res->code = err;
        req->res->error_message = uv_strerror(err);
        Log::Warning(Event::HttpRequest, err, url + ": " + uv_strerror(err));
        req->complete();
        return req;
    }
    uv_file fd = err;

    uv_fs_t stat_req;
    uv_fs_fstat(l, &stat_req, fd, nullptr);
    uv_fs_req_cleanup(&stat_req);
    err = uv_fs_fstat(l, &stat_req, fd, nullptr);
    if (err < 0) {
        req->res->code = err;
        req->res->error_message = uv_strerror(err);
        Log::Warning(Event::HttpRequest, err, url + ": " + uv_strerror(err));
        req->complete();
        return req;
    }

    const uint64_t size = static_cast<const uv_stat_t*>(stat_req.ptr)->st_size;


    std::string body;
    body.resize(size);
    uv_buf_t uvbuf = uv_buf_init(const_cast<char *>(body.data()), body.size());

    uv_fs_t read_req;
    err = uv_fs_read(l, &read_req, fd, &uvbuf, 1, 0, nullptr);
    uv_fs_req_cleanup(&read_req);
    if (err < 0) {
        req->res->code = err;
        req->res->error_message = uv_strerror(err);
        Log::Warning(Event::HttpRequest, err, url + ": " + uv_strerror(err));
        req->complete();
        return req;
    }


    uv_fs_t close_req;
    err = uv_fs_close(l, &close_req, fd, nullptr);
    uv_fs_req_cleanup(&close_req);
    if (err < 0) {
        req->res->code = err;
        req->res->error_message = uv_strerror(err);
        Log::Warning(Event::HttpRequest, err, url + ": " + uv_strerror(err));
        req->complete();
        return req;
    }

    req->res->body.swap(body);
    req->res->code = 200;
    Log::Info(Event::HttpRequest, 200, url);
    req->complete();

    return req;
}

void platform::cancel_request_http(const std::shared_ptr<Request> &req) {
    if (req) {
        req->cancelled = true;
    }
}

} // end namespace llmr