summaryrefslogtreecommitdiff
blob: 83d2e5ff34288799df42f245eb3840c400d0e400 (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
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
-module(lgn_sign_in).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% TYPES %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
-record
(
   input,
   {
      username :: binary(),
      password :: binary()
   }
).

-record
(
   query_state,
   {
      player_id :: shr_player:id(),
      player :: shr_player:type()
   }
).

-type input() :: #input{}.
-type query_state() :: #query_state{}.

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% EXPORTS %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
-export([out/1]).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% LOCAL FUNCTIONS %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
-spec parse_input (shr_query:type()) -> input().
parse_input (Query) ->
   JSONReqMap = shr_query:get_params(Query),
   Username = maps:get(<<"usr">>, JSONReqMap),
   Password = maps:get(<<"pwd">>, JSONReqMap),

   #input
   {
      username = string:lowercase(Username),
      password = Password
   }.

-spec fetch_data (input()) -> query_state().
fetch_data (Input) ->
   Username = Input#input.username,

   % Having this be cached my be both useless and a security issue.
   PlayerID = shr_timed_cache:fetch(login_db, ataxia_security:any(), Username),

   Player = shr_timed_cache:fetch(player_db, ataxia_security:any(), PlayerID),

   #query_state
   {
      player_id = PlayerID,
      player = Player
   }.

-spec update_data (query_state(), input()) -> query_state().
update_data (QueryState, Input) ->
   InputPassword = Input#input.password,
   Player = QueryState#query_state.player,

   true = (shr_player:password_is(InputPassword, Player)),

   S0Player = shr_player:new_token(Player),
   S1Player = shr_player:refresh_active(S0Player),

   QueryState#query_state
   {
      player = S1Player
   }.

-spec commit_update (query_state()) -> 'ok'.
commit_update (QueryState) ->
   PlayerID = QueryState#query_state.player_id,
   UpdatedPlayer = QueryState#query_state.player,
   NewToken = shr_player:get_token(UpdatedPlayer),
   NewActiveTime = shr_player:get_last_active(UpdatedPlayer),

   ok =
      ataxia_client:update
      (
         player_db,
         ataxia_security:user_from_id(PlayerID),
         ataxic:update_value
         (
            ataxic:sequence
            (
               [
                  ataxic:update_field
                  (
                     shr_player:get_token_field(),
                     ataxic:constant(NewToken)
                  ),
                  ataxic:update_field
                  (
                     shr_player:get_last_active_field(),
                     ataxic:constant(NewActiveTime)
                  )
               ]
            )
         ),
         PlayerID
      ),

   shr_timed_cache:update
   (
      player_db,
      ataxia_security:any(),
      PlayerID,
      UpdatedPlayer
   ),

   'ok'.

-spec generate_reply(query_state()) -> binary().
generate_reply (QueryState) ->
   Player = QueryState#query_state.player,
   PlayerID = QueryState#query_state.player_id,

   SetSession = lgn_set_session:generate(PlayerID, Player),
   Output = jiffy:encode([SetSession]),

   Output.

-spec handle (shr_query:type()) -> binary().
handle (Query) ->
   Input = parse_input(Query),
   QueryState = fetch_data(Input),
   Update = update_data(QueryState, Input),
   commit_update(Update),
   generate_reply(Update).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% EXPORTED FUNCTIONS %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
out(A) ->
   {
      content,
      "application/json; charset=UTF-8",
      handle(shr_query:new(A))
   }.