Renderer.pm 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. package Trog::Renderer;
  2. use strict;
  3. use warnings;
  4. no warnings 'experimental';
  5. use feature qw{signatures state};
  6. use Carp::Always;
  7. use Trog::Vars;
  8. use Trog::Log qw{:all};
  9. use Trog::Renderer::text;
  10. use Trog::Renderer::html;
  11. use Trog::Renderer::json;
  12. use Trog::Renderer::blob;
  13. use Trog::Renderer::css;
  14. =head1 Trog::Renderer
  15. Idea here is to have a renderer per known/supported content-type we need to output that is also theme-aware.
  16. We have an abstraction here, render() which you feed everything to.
  17. =cut;
  18. our %renderers = (
  19. text => \&Trog::Renderer::text::render,
  20. html => \&Trog::Renderer::html::render,
  21. json => \&Trog::Renderer::json::render,
  22. blob => \&Trog::Renderer::blob::render,
  23. xsl => \&Trog::Renderer::text::render,
  24. xml => \&Trog::Renderer::text::render,
  25. css => \&Trog::Renderer::css::render,
  26. );
  27. =head2 Trog::Renderer->render(%options)
  28. Returns either the 3-arg arrayref suitable to emit at the end of a PSGI session or a response body if the component field of options is truthy.
  29. The idea is that components will be concatenated to other rendered templates until we finish having everything ready.
  30. =cut
  31. sub render ($class, %options) {
  32. local $@;
  33. my $renderer;
  34. return _yeet($renderer, "Renderer requires a valid content type to be passed", %options) unless $options{contenttype};
  35. my $rendertype = $Trog::Vars::byct{$options{contenttype}};
  36. return _yeet($renderer, "Renderer requires a known content type (used $options{contenttype}) to be passed", %options) unless $rendertype;
  37. $renderer = $renderers{$rendertype};
  38. return _yeet($renderer, "Renderer for $rendertype is not defined!", %options) unless $renderer;
  39. return _yeet($renderer, "Status code not provided", %options) if !$options{code} && !$options{component};
  40. return _yeet($renderer, "Template data not provided", %options) unless $options{data};
  41. return _yeet($renderer, "Template not provided", %options) unless $options{template};
  42. my $skip_save = !$options{data}{route} || $options{data}{has_query} || $options{data}{user} || ($options{code} // 0) != 200 || Trog::Log::is_debug();
  43. my $ret;
  44. local $@;
  45. eval {
  46. $ret = $renderer->(%options);
  47. save_render( $options{data}, $ret->[2], $ret->[1]) unless $skip_save;
  48. 1;
  49. } or do {
  50. return _yeet($renderer, $@, %options);
  51. };
  52. return $ret;
  53. }
  54. sub _yeet ($renderer, $error, %options) {
  55. WARN($error);
  56. # All-else fails error page
  57. my $ret;
  58. local $@;
  59. eval {
  60. $ret = $renderer->(
  61. code => 500,
  62. template => '500.tx',
  63. contenttype => 'text/html',
  64. data => { %options, content => "<h1>500 Internal Server Error</h1>$error" },
  65. );
  66. 1;
  67. } or do {
  68. my $msg = $error;
  69. $msg .= " and subsequently during render of error template, $@" if $renderer;
  70. INFO("$options{data}{method} 500 $options{data}{route}");
  71. FATAL($msg);
  72. };
  73. return $ret;
  74. }
  75. sub save_render ( $vars, $body, %headers ) {
  76. Path::Tiny::path( "www/statics/" . dirname( $vars->{route} ) )->mkpath;
  77. my $file = "www/statics/$vars->{route}";
  78. my $verb = -f $file ? 'Overwrite' : 'Write';
  79. DEBUG("$verb static for $vars->{route}");
  80. open( my $fh, '>', $file ) or die "Could not open $file for writing";
  81. print $fh "HTTP/1.1 $vars->{code} OK\n";
  82. foreach my $h ( keys(%headers) ) {
  83. print $fh "$h:$headers{$h}\n" if $headers{$h};
  84. }
  85. print $fh "\n";
  86. print $fh $body;
  87. close $fh;
  88. }
  89. 1;