summaryrefslogtreecommitdiff
path: root/src/mem3/src/mem3_httpd.erl
blob: 745fe815ca6b07b6fa650f64b18ec0938b431cb6 (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
107
108
109
110
111
112
113
114
115
116
117
118
% 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.

-module(mem3_httpd).

-export([
    handle_membership_req/1,
    handle_shards_req/2,
    handle_sync_req/2
]).

%% includes
-include_lib("mem3/include/mem3.hrl").
-include_lib("couch/include/couch_db.hrl").

handle_membership_req(
    #httpd{
        method = 'GET',
        path_parts = [<<"_membership">>]
    } = Req
) ->
    ClusterNodes =
        try
            mem3:nodes()
        catch
            _:_ -> {ok, []}
        end,
    couch_httpd:send_json(
        Req,
        {[
            {all_nodes, lists:sort([node() | nodes()])},
            {cluster_nodes, lists:sort(ClusterNodes)}
        ]}
    );
handle_membership_req(#httpd{path_parts = [<<"_membership">>]} = Req) ->
    chttpd:send_method_not_allowed(Req, "GET").

handle_shards_req(
    #httpd{
        method = 'GET',
        path_parts = [_DbName, <<"_shards">>]
    } = Req,
    Db
) ->
    DbName = mem3:dbname(couch_db:name(Db)),
    Shards = mem3:shards(DbName),
    JsonShards = json_shards(Shards, dict:new()),
    couch_httpd:send_json(
        Req,
        {[
            {shards, JsonShards}
        ]}
    );
handle_shards_req(
    #httpd{
        method = 'GET',
        path_parts = [_DbName, <<"_shards">>, DocId]
    } = Req,
    Db
) ->
    DbName = mem3:dbname(couch_db:name(Db)),
    Shards = mem3:shards(DbName, DocId),
    {[{Shard, Dbs}]} = json_shards(Shards, dict:new()),
    couch_httpd:send_json(
        Req,
        {[
            {range, Shard},
            {nodes, Dbs}
        ]}
    );
handle_shards_req(#httpd{path_parts = [_DbName, <<"_shards">>]} = Req, _Db) ->
    chttpd:send_method_not_allowed(Req, "GET");
handle_shards_req(#httpd{path_parts = [_DbName, <<"_shards">>, _DocId]} = Req, _Db) ->
    chttpd:send_method_not_allowed(Req, "GET").

handle_sync_req(
    #httpd{
        method = 'POST',
        path_parts = [_DbName, <<"_sync_shards">>]
    } = Req,
    Db
) ->
    DbName = mem3:dbname(couch_db:name(Db)),
    ShardList = [S#shard.name || S <- mem3:ushards(DbName)],
    [sync_shard(S) || S <- ShardList],
    chttpd:send_json(Req, 202, {[{ok, true}]});
handle_sync_req(Req, _) ->
    chttpd:send_method_not_allowed(Req, "POST").

%%
%% internal
%%

json_shards([], AccIn) ->
    List = dict:to_list(AccIn),
    {lists:sort(List)};
json_shards([#shard{node = Node, range = [B, E]} | Rest], AccIn) ->
    HexBeg = couch_util:to_hex(<<B:32/integer>>),
    HexEnd = couch_util:to_hex(<<E:32/integer>>),
    Range = list_to_binary(HexBeg ++ "-" ++ HexEnd),
    json_shards(Rest, dict:append(Range, Node, AccIn)).

sync_shard(ShardName) ->
    Shards = mem3_shards:for_shard_range(ShardName),
    [
        rpc:call(S1#shard.node, mem3_sync, push, [S1, S2#shard.node])
     || S1 <- Shards, S2 <- Shards, S1 =/= S2
    ],
    ok.