#include "webfuse/provider/impl/operation/lookup.h"
#include "webfuse/mocks/mock_request.hpp"
#include "webfuse/mocks/mock_provider.hpp"
#include "webfuse/mocks/fake_invokation_context.hpp"

#include <gtest/gtest.h>
#include <cstdlib>

using ::webfuse_test::MockProvider;
using ::webfuse_test::MockRequest;
using ::webfuse_test::StatMatcher;
using ::webfuse_test::create_context;
using ::testing::_;
using ::testing::Invoke;
using ::testing::StrEq;

namespace
{

void free_request(wfp_request * request, ino_t, char const *)
{
    free(request);
}

}

TEST(wfp_impl_lookup, invoke_provider)
{
    ino_t inode = 42;
    MockProvider provider;
    EXPECT_CALL(provider,lookup(_, inode,StrEq("some.file"))).Times(1)
        .WillOnce(Invoke(free_request));

    wfp_request request = {nullptr, nullptr, 0};
    wfp_impl_invokation_context context = create_context(provider, &request);

    json_t * params = json_array();
    json_array_append_new(params, json_string("test.filesystem"));
    json_array_append_new(params, json_integer(inode));
    json_array_append_new(params, json_string("some.file"));

    wfp_impl_lookup(&context, params, 42);
    json_decref(params);
}

TEST(wfp_impl_lookup, fail_invalid_param_count)
{
    MockProvider provider;
    EXPECT_CALL(provider,lookup(_, _,_)).Times(0);

    wfp_request request = {nullptr, nullptr, 0};
    wfp_impl_invokation_context context = create_context(provider, &request);

    json_t * params = json_array();
    json_array_append_new(params, json_string("test.filesystem"));
    json_array_append_new(params, json_integer(23));

    wfp_impl_lookup(&context, params, 42);
    json_decref(params);
}

TEST(wfp_impl_lookup, fail_invalid_inode_type)
{
    MockProvider provider;
    EXPECT_CALL(provider,lookup(_, _,_)).Times(0);

    wfp_request request = {nullptr, nullptr, 0};
    wfp_impl_invokation_context context = create_context(provider, &request);

    json_t * params = json_array();
    json_array_append_new(params, json_string("test.filesystem"));
    json_array_append_new(params, json_string("23"));
    json_array_append_new(params, json_string("some.file"));

    wfp_impl_lookup(&context, params, 42);
    json_decref(params);
}

TEST(wfp_impl_lookup, fail_invalid_name_type)
{
    MockProvider provider;
    EXPECT_CALL(provider,lookup(_, _,_)).Times(0);

    wfp_request request = {nullptr, nullptr, 0};
    wfp_impl_invokation_context context = create_context(provider, &request);

    json_t * params = json_array();
    json_array_append_new(params, json_string("test.filesystem"));
    json_array_append_new(params, json_integer(23));
    json_array_append_new(params, json_integer(1));

    wfp_impl_lookup(&context, params, 42);
    json_decref(params);
}

TEST(wfp_impl_lookup, default_responds_error)
{
    MockRequest request;
    auto * req = request.create_request(42);
    EXPECT_CALL(request, respond_error(_,42)).Times(1);

    wfp_impl_lookup_default(req, 1, "some.file", nullptr);
}

TEST(wfp_impl_lookup, respond_file)
{
    MockRequest request;
    auto * req = request.create_request(42);
    EXPECT_CALL(request, respond(StatMatcher(23, 0754, "file"), 42)).Times(1);

    struct stat buffer;
    memset(&buffer, 0, sizeof(buffer));
    buffer.st_ino = 23;
    buffer.st_mode = S_IFREG | 0754;
    wfp_impl_respond_lookup(req, &buffer);
}

TEST(wfp_impl_lookup, respond_dir)
{
    MockRequest request;
    auto * req = request.create_request(42);
    EXPECT_CALL(request, respond(StatMatcher(23, 0754, "dir"), 42)).Times(1);

    struct stat buffer;
    memset(&buffer, 0, sizeof(buffer));
    buffer.st_ino = 23;
    buffer.st_mode = S_IFDIR | 0754;
    wfp_impl_respond_lookup(req, &buffer);
}