blob: e7fc5188acc0b96f9eb649f023d71daad66bf590 [file] [log] [blame]
Akron0e1ed242018-10-11 13:22:00 +02001#!/usr/bin/env perl
2use Mojolicious::Lite;
3use Mojo::ByteStream 'b';
4use Mojo::Date;
5use Mojo::JSON qw/true false encode_json decode_json/;
6use strict;
7use warnings;
8use Mojo::JWT;
Akron6d49c1f2018-10-11 14:22:21 +02009use Mojo::File qw/path/;
10use Mojo::Util qw/slugify/;
Akron0e1ed242018-10-11 13:22:00 +020011
12# This is an API fake server with fixtures
13
14my $secret = 's3cr3t';
Akron6d49c1f2018-10-11 14:22:21 +020015my $fixture_path = path(Mojo::File->new(__FILE__)->dirname);
Akron0e1ed242018-10-11 13:22:00 +020016
17helper jwt_encode => sub {
18 shift;
19 return Mojo::JWT->new(
20 secret => $secret,
21 token_type => 'api_token',
22 expires => time + (3 * 34 * 60 * 60),
23 claims => { @_ }
24 );
25};
26
27helper jwt_decode => sub {
28 my ($c, $auth) = @_;
29 $auth =~ s/\s*api_token\s+//;
30 return Mojo::JWT->new(secret => $secret)->decode($auth);
31};
32
33
Akron6d49c1f2018-10-11 14:22:21 +020034# Load fixture responses
35helper 'load_response' => sub {
36 my $c = shift;
37 my $q_name = shift;
38 my $file = $fixture_path->child("response_$q_name.json");
39 unless (-f $file) {
40 return {
41 status => 500,
42 json => {
43 errors => [[0, 'Unable to load query response from ' . $file]]
44 }
45 }
46 };
47 my $response = $file->slurp;
48 return decode_json($response);
49};
50
51
Akron0e1ed242018-10-11 13:22:00 +020052# Base page
53get '/' => sub {
Akron6d49c1f2018-10-11 14:22:21 +020054 shift->render(text => 'Fake server available');
Akron0e1ed242018-10-11 13:22:00 +020055};
56
Akron32396632018-10-11 17:08:37 +020057
Akron0e1ed242018-10-11 13:22:00 +020058# Search fixtures
59get '/search' => sub {
60 my $c = shift;
61 my $v = $c->validation;
62 $v->optional('q');
63 $v->optional('page');
64 $v->optional('ql');
65 $v->optional('count');
66 $v->optional('context');
67
Akron32396632018-10-11 17:08:37 +020068 $c->app->log->debug('Receive request');
69
Akron0e1ed242018-10-11 13:22:00 +020070 # Response q=x&ql=cosmas3
71 if ($v->param('ql') && $v->param('ql') eq 'cosmas3') {
72 return $c->render(
73 status => 400,
74 json => {
75 "\@context" => "http://korap.ids-mannheim.de/ns/koral/0.3/context.jsonld",
76 "errors" => [[307,"cosmas3 is not a supported query language!"]]
77 });
78 };
79
Akron6d49c1f2018-10-11 14:22:21 +020080 if (!$v->param('q')) {
81 return $c->render(%{$c->load_response('no_query')});
Akron0e1ed242018-10-11 13:22:00 +020082 };
83
Akron6d49c1f2018-10-11 14:22:21 +020084 # Get response based on query parameter
85 my $response = $c->load_response(slugify($v->param('q')));
Akron0e1ed242018-10-11 13:22:00 +020086
87 # Check authentification
88 if (my $auth = $c->req->headers->header('Authorization')) {
89 if (my $jwt = $c->jwt_decode($auth)) {
Akron6d49c1f2018-10-11 14:22:21 +020090 $response->{json}->{meta}->{authorized} = $jwt->{username} if $jwt->{username};
Akron0e1ed242018-10-11 13:22:00 +020091 };
92 };
93
Akron6d49c1f2018-10-11 14:22:21 +020094 # Set page parameter
Akron0e1ed242018-10-11 13:22:00 +020095 if ($v->param('page')) {
Akron6d49c1f2018-10-11 14:22:21 +020096 $response->{json}->{meta}->{startIndex} = $v->param("startIndex");
Akron0e1ed242018-10-11 13:22:00 +020097 };
98
Akron0e1ed242018-10-11 13:22:00 +020099 # Simple search fixture
Akron32396632018-10-11 17:08:37 +0200100 $c->render(%$response);
101
102 $c->app->log->debug('Rendered result');
103
104 return 1;
Akron0e1ed242018-10-11 13:22:00 +0200105};
106
107
108
109############
110# Auth API #
111############
112
113# Request API token
114get '/auth/logout' => sub {
115 my $c = shift;
116
117 if (my $auth = $c->req->headers->header('Authorization')) {
118 if (my $jwt = $c->jwt_decode($auth)) {
119 my $user = $jwt->{username} if $jwt->{username};
120
121 $c->app->log->debug('Server-Logout: ' . $user);
122 return $c->render(json => { msg => [[0, 'Fine!']]});
123 };
124 };
125
126 return $c->render(status => 400, json => { error => [[0, 'No!']]});
127};
128
129
130# Request API token
131get '/auth/apiToken' => sub {
132 my $c = shift;
133
134 # Get auth header
135 my $auth = $c->req->headers->authorization;
136
137 # Authorization missing or not basic
138 if (!$auth || $auth !~ s/\s*Basic\s+//gi) {
139 return $c->render(
140 json => {
141 error => [[2, 'x']]
142 }
143 );
144 };
145
146 # Decode header
147 my ($username, $pwd) = @{b($auth)->b64_decode->split(':')->to_array};
148
149 # the password is 'pass'
150 if ($pwd) {
151
152 # the password is 'pass'
153 if ($pwd eq 'pass') {
154
155 # Render info with token
156 my $jwt = $c->jwt_encode(username => $username);
157
158 # Render in the Kustvakt fashion:
159 return $c->render(
160 format => 'html',
161 text => encode_json({
162 %{$jwt->claims},
163 expires => $jwt->expires,
164 token => $jwt->encode,
165 token_type => 'api_token'
166 })
167 );
168 };
169
170 return $c->render(
171 json => {
172 error => [[2004, undef]]
173 }
174 );
175 };
176
177 return $c->render(
178 json => {
179 error => [[2004, undef]]
180 }
181 );
182};
183
184app->start;
185
186
187__END__
188
189
190 # Temporary:
191 my $collection_query = {
192 '@type' => "koral:docGroup",
193 "operation" => "operation:or",
194 "operands" => [
195 {
196 '@type' => "koral:docGroup",
197 "operation" => "operation:and",
198 "operands" => [
199 {
200 '@type' => "koral:doc",
201 "key" => "title",
202 "match" => "match:eq",
203 "value" => "Der Birnbaum",
204 "type" => "type:string"
205 },
206 {
207 '@type' => "koral:doc",
208 "key" => "pubPlace",
209 "match" => "match:eq",
210 "value" => "Mannheim",
211 "type" => "type:string"
212 },
213 {
214 '@type' => "koral:docGroup",
215 "operation" => "operation:or",
216 "operands" => [
217 {
218 '@type' => "koral:doc",
219 "key" => "subTitle",
220 "match" => "match:eq",
221 "value" => "Aufzucht oder Pflege",
222 "type" => "type:string"
223 },
224 {
225 '@type' => "koral:doc",
226 "key" => "subTitle",
227 "match" => "match:eq",
228 "value" => "Gedichte",
229 "type" => "type:string"
230 }
231 ]
232 }
233 ]
234 },
235 {
236 '@type' => "koral:doc",
237 "key" => "pubDate",
238 "match" => "match:geq",
239 "value" => "2015-03-05",
240 "type" => "type:date"
241 }
242 ]
243 };