///
/// Copyright 2019 Oliver Giles
///
/// This file is part of Laminar
///
/// Laminar is free software: you can redistribute it and/or modify
/// it under the terms of the GNU General Public License as published by
/// the Free Software Foundation, either version 3 of the License, or
/// (at your option) any later version.
///
/// Laminar is distributed in the hope that it will be useful,
/// but WITHOUT ANY WARRANTY; without even the implied warranty of
/// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
/// GNU General Public License for more details.
///
/// You should have received a copy of the GNU General Public License
/// along with Laminar.  If not, see <http://www.gnu.org/licenses/>
///
#ifndef LAMINAR_EVENTSOURCE_H_
#define LAMINAR_EVENTSOURCE_H_

#include <kj/async-io.h>
#include <kj/compat/http.h>
#include <rapidjson/document.h>
#include <vector>

class EventSource {
public:
    EventSource(kj::AsyncIoContext& ctx, const char* httpConnectAddr, const char* path) :
        networkAddress(ctx.provider->getNetwork().parseAddress(httpConnectAddr).wait(ctx.waitScope)),
        httpClient(kj::newHttpClient(ctx.lowLevelProvider->getTimer(), headerTable, *networkAddress)),
        headerTable(),
        headers(headerTable),
        buffer(kj::heapArrayBuilder<char>(BUFFER_SIZE))
    {
        headers.add("Accept", "text/event-stream");
        auto resp = httpClient->request(kj::HttpMethod::GET, path, headers).response.wait(ctx.waitScope);
        promise = waitForMessages(resp.body.get(), 0).attach(kj::mv(resp));
    }

    const std::vector<rapidjson::Document>& messages() {
        return receivedMessages;
    }

private:
    kj::Own<kj::NetworkAddress> networkAddress;
    kj::Own<kj::HttpClient> httpClient;
    kj::HttpHeaderTable headerTable;
    kj::HttpHeaders headers;
    kj::ArrayBuilder<char> buffer;
    kj::Maybe<kj::Promise<void>> promise;
    std::vector<rapidjson::Document> receivedMessages;

    kj::Promise<void> waitForMessages(kj::AsyncInputStream* stream, ulong offset) {
        return stream->read(buffer.asPtr().begin() + offset, 1, BUFFER_SIZE).then([=](size_t s) {
            ulong end = offset + s;
            buffer.asPtr().begin()[end] = '\0';
            if(strcmp(&buffer.asPtr().begin()[end - 2], "\n\n") == 0) {
                rapidjson::Document d;
                d.Parse(buffer.begin() + strlen("data: "));
                receivedMessages.emplace_back(kj::mv(d));
                end = 0;
            }
            return waitForMessages(stream, end);
        });
    }

    static const int BUFFER_SIZE = 1024;
};

#endif // LAMINAR_EVENTSOURCE_H_