JSON.pm 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. package Trog::Routes::JSON;
  2. use strict;
  3. use warnings;
  4. no warnings 'experimental';
  5. use feature qw{signatures state};
  6. use Clone qw{clone};
  7. use JSON::MaybeXS();
  8. use Trog::Config();
  9. use Trog::Auth();
  10. use Trog::Routes::HTML();
  11. my $conf = Trog::Config::get();
  12. # TODO de-duplicate this, it's shared in html
  13. my $theme_dir = '';
  14. $theme_dir = "themes/" . $conf->param('general.theme') if $conf->param('general.theme') && -d "www/themes/" . $conf->param('general.theme');
  15. our %routes = (
  16. '/api/catalog' => {
  17. method => 'GET',
  18. callback => \&catalog,
  19. parameters => [],
  20. },
  21. '/api/webmanifest' => {
  22. method => 'GET',
  23. callback => \&webmanifest,
  24. parameters => [],
  25. },
  26. '/api/version' => {
  27. method => 'GET',
  28. callback => \&version,
  29. parameters => [],
  30. },
  31. '/api/auth_change_request/(.*)' => {
  32. method => 'GET',
  33. callback => \&process_auth_change_request,
  34. captures => ['token'],
  35. noindex => 1,
  36. },
  37. );
  38. # Clone / redact for catalog
  39. my $cloned = clone( \%routes );
  40. foreach my $r ( keys(%$cloned) ) {
  41. delete $cloned->{$r}{callback};
  42. }
  43. my $enc = JSON::MaybeXS->new( utf8 => 1 );
  44. # Note to authors, don't forget to update this
  45. sub _version () {
  46. return '1.0';
  47. }
  48. # Special case of a non data-structure JSON return
  49. sub version ($query) {
  50. state $ret = [ 200, [ 'Content-type' => "application/json", ETag => 'version-' . _version() ], [ _version() ] ];
  51. return $ret;
  52. }
  53. sub catalog ($query) {
  54. return _render(200, { ETag => 'catalog-' . _version() }, %$cloned);
  55. }
  56. sub webmanifest ($query) {
  57. state $headers = { ETag => 'manifest-' . _version() };
  58. state %manifest = (
  59. "icons" => [
  60. { "src" => "$theme_dir/img/icon/favicon-32.png", "type" => "image/png", "sizes" => "32x32" },
  61. { "src" => "$theme_dir/img/icon/favicon-48.png", "type" => "image/png", "sizes" => "48x48" },
  62. { "src" => "$theme_dir/img/icon/favicon-167.png", "type" => "image/png", "sizes" => "167x167" },
  63. { "src" => "$theme_dir/img/icon/favicon-180.png", "type" => "image/png", "sizes" => "180x180" },
  64. { "src" => "$theme_dir/img/icon/favicon-192.png", "type" => "image/png", "sizes" => "192x192" },
  65. { "src" => "$theme_dir/img/icon/favicon-512.png", "type" => "image/png", "sizes" => "512x512" },
  66. ],
  67. );
  68. return _render(200, $headers, %manifest );
  69. }
  70. sub process_auth_change_request($query) {
  71. my $token = $query->{token};
  72. my $msg = Trog::Auth::process_change_request($token);
  73. return Trog::Routes::HTML::forbidden($query) unless $msg;
  74. return _render(200, undef,
  75. message => $msg,
  76. result => 'success',
  77. );
  78. }
  79. sub _render ($code, $headers, %data) {
  80. return Trog::Renderer->render(
  81. code => 200,
  82. data => \%data,
  83. template => 'bogus.tx',
  84. contenttype => 'application/json',
  85. headers => $headers,
  86. );
  87. }
  88. 1;