rexyai/RestRserve

Static Route fails to return csv, rds, jpg, pdf (png and txt works) for folder

Closed this issue · 5 comments

Bug

Serving files via a static endpoint results in an encoding error depending on content-type.
csv, rds, jpg, and pdf fail, with: can't encode body with content_type = 'image/{TYPE}' .
png images and .txt files work however.

Reproduction

################################################################################
# Setup the static data directory and write some example files

if (!dir.exists("static")) dir.create("static")
set.seed(123)
png("static/testplot.png")
plot(1:100, cumsum(rnorm(100)), type = "l")
dev.off()
jpeg("static/testplot.jpg")
plot(1:100, cumsum(rnorm(100)), type = "l")
dev.off()
pdf("static/testplot.pdf")
plot(1:100, cumsum(rnorm(100)), type = "l")
dev.off()

writeLines("Hello World", "static/test.txt")
writeLines("print('Hello World')", "static/test.R")
write.csv(data.frame(x = 1:100, y = cumsum(rnorm(100))), "static/testdata.csv")
saveRDS(data.frame(x = 1:100, y = cumsum(rnorm(100))), "static/testdata.rds")


################################################################################
# Define the webserver
library(RestRserve)
app = Application$new()
app$logger$set_log_level("debug")

app$add_get(path = "/ping", FUN = function(.req, .res) .res$set_body("OK"))
app$add_get(path = "/list_files", FUN = function(.req, .res) .res$set_body(list.files("static")))
app$add_static("/static", "static")


# define a quick helper function to make testing the endpoints easier
test_endpoint = function(end) {
  req = Request$new(path = end, method = "GET")
  cat(app$process_request(req)$body, "\n")
}

################################################################################
## First check if the basic methodology works: ALL GOOD

# ping works
test_endpoint("/ping")
#> {"timestamp":"2022-03-16 10:09:54.865609","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"d8494ad4-a508-11ec-8000-f398b9230932","request":{"method":"GET","path":"/ping","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 10:09:54.865609","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"d8494ad4-a508-11ec-8000-f398b9230932","response":{"status_code":200,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> OK 

# all files are found
test_endpoint("/list_files")
#> {"timestamp":"2022-03-16 09:43:42.430613","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"2ef14aca-a505-11ec-8000-f398b9230932","request":{"method":"GET","path":"/list_files","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 09:43:42.431622","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"2ef14aca-a505-11ec-8000-f398b9230932","response":{"status_code":200,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> test.R
#> test.txt
#> testdata.csv
#> testdata.rds
#> testplot.jpg
#> testplot.pdf
#> testplot.png 


################################################################################
## Second, check if all files can be downloaded. Works for png, R, and txt but not the others...

# R works
test_endpoint("/static/test.R")
#> {"timestamp":"2022-03-16 09:58:32.781083","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"3f5f3ca8-a507-11ec-8000-f398b9230932","request":{"method":"GET","path":"/static/test.txt","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 09:58:32.782011","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"3f5f3ca8-a507-11ec-8000-f398b9230932","response":{"status_code":200,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> C:\Users\REDACTED\test-restrserve\static//test.R

# TXT works
test_endpoint("/static/test.txt")
#> {"timestamp":"2022-03-16 09:58:32.781083","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"3f5f3ca8-a507-11ec-8000-f398b9230932","request":{"method":"GET","path":"/static/test.txt","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 09:58:32.782011","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"3f5f3ca8-a507-11ec-8000-f398b9230932","response":{"status_code":200,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> C:\Users\REDACTED\test-restrserve\static//test.txt


# CSV fails
test_endpoint("/static/testdata.csv")
#> {"timestamp":"2022-03-16 09:51:30.174911","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"435fa41a-a506-11ec-8000-f398b9230932","request":{"method":"GET","path":"/static/testdata.csv","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 09:51:30.176912","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"435fa41a-a506-11ec-8000-f398b9230932","response":{"status_code":500,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> 500 Internal Server Error: can't encode body with content_type = 'text/csv' 


# RDS (octet-stream) fails
test_endpoint("/static/testdata.rds")
#> {"timestamp":"2022-03-16 09:52:04.413598","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"5a255096-a506-11ec-8000-f398b9230932","request":{"method":"GET","path":"/static/testdata.rds","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 09:52:04.414597","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"5a255096-a506-11ec-8000-f398b9230932","response":{"status_code":500,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> 500 Internal Server Error: can't encode body with content_type = 'application/octet-stream'


# PNG works as expected
test_endpoint("/static/testplot.png")
#> {"timestamp":"2022-03-16 09:44:17.175594","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"437f2b1a-a505-11ec-8000-f398b9230932","request":{"method":"GET","path":"/static/testplot.png","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 09:44:17.177634","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"437f2b1a-a505-11ec-8000-f398b9230932","response":{"status_code":200,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> C:\Users\REDACTED\test-restrserve\static//testplot.png 


# PDF fails
test_endpoint("/static/testplot.pdf")
#> {"timestamp":"2022-03-16 09:44:46.284985","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"54d8f580-a505-11ec-8000-f398b9230932","request":{"method":"GET","path":"/static/testplot.pdf","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 09:44:46.286984","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"54d8f580-a505-11ec-8000-f398b9230932","response":{"status_code":500,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> 500 Internal Server Error: can't encode body with content_type = 'application/pdf' 


# JPG fails
test_endpoint("/static/testplot.jpg")
#> {"timestamp":"2022-03-16 09:45:02.932342","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"5eced78a-a505-11ec-8000-f398b9230932","request":{"method":"GET","path":"/static/testplot.jpg","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 09:45:02.934370","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"5eced78a-a505-11ec-8000-f398b9230932","response":{"status_code":500,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> 500 Internal Server Error: can't encode body with content_type = 'image/jpeg'


################################################################################
# Check if the error also happens if the endpoint only serves a single file

app$add_static("/csv", "static/testdata.csv")

# also fails for serving single files
test_endpoint("/csv")
#> {"timestamp":"2022-03-16 10:03:42.505070","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"fa57bc56-a507-11ec-8000-f398b9230932","request":{"method":"GET","path":"/csv","parameters_query":[],"parameters_path":[],"headers":{}}}}
#> {"timestamp":"2022-03-16 10:03:42.507071","level":"DEBUG","name":"Application","pid":22568,"msg":"","context":{"request_id":"fa57bc56-a507-11ec-8000-f398b9230932","response":{"status_code":500,"headers":{"Server":"RestRserve/0.4.1"}}}}
#> 500 Internal Server Error: can't encode body with content_type = 'text/csv'

Expected Behavior

Should return CSV, RDS, JPG, and PDF files as it returns PNG and TXT files.

Environment information

Output of sessioninfo::session_info()

- Session info ------------------------------------------------------------------------
 setting  value
 version  R version 4.1.2 (2021-11-01)
 os       Windows 10 x64 (build 19043)
 system   x86_64, mingw32
 ui       RStudio
 language (EN)
 collate  German_Germany.1252
 ctype    German_Germany.1252
 tz       Europe/Berlin
 date     2022-03-16
 rstudio  2021.09.2+382 Ghost Orchid (desktop)
 pandoc   2.17.1.1 @ C:/Users/REDACTED/AppData/Local/Programs/Quarto/bin/ (via rmarkdown)

- Packages ----------------------------------------------------------------------------
 package     * version date (UTC) lib source
 backports     1.4.1   2021-12-13 [1] CRAN (R 4.1.2)
 checkmate     2.0.0   2020-02-06 [1] CRAN (R 4.1.2)
 cli           3.1.0   2021-10-27 [1] CRAN (R 4.1.2)
 digest        0.6.29  2021-12-01 [1] CRAN (R 4.1.2)
 evaluate      0.14    2019-05-28 [1] CRAN (R 4.1.2)
 fastmap       1.1.0   2021-01-25 [1] CRAN (R 4.1.2)
 htmltools     0.5.2   2021-08-25 [1] CRAN (R 4.1.2)
 jsonlite      1.8.0   2022-02-22 [1] CRAN (R 4.1.2)
 knitr         1.37    2021-12-16 [1] CRAN (R 4.1.2)
 mime          0.12    2021-09-28 [1] CRAN (R 4.1.1)
 R6            2.5.1   2021-08-19 [1] CRAN (R 4.1.2)
 Rcpp          1.0.8   2022-01-13 [1] CRAN (R 4.1.2)
 RestRserve  * 0.4.1   2021-01-04 [1] CRAN (R 4.1.2)
 rlang         0.4.12  2021-10-18 [1] CRAN (R 4.1.2)
 rmarkdown     2.11    2021-09-14 [1] CRAN (R 4.1.2)
 sessioninfo   1.2.2   2021-12-06 [1] CRAN (R 4.1.2)
 uuid          1.0-3   2021-11-01 [1] CRAN (R 4.1.2)
 xfun          0.29    2021-12-14 [1] CRAN (R 4.1.2)
 yaml          2.2.1   2020-02-01 [1] CRAN (R 4.1.1)

 [1] C:/Users/REDACTED/Documents/R/R-4.1.2/library

---------------------------------------------------------------------------------------

Outro

Having said that, thank you all so much for this wonderful library, really makes my life easier and is highly appreciated!

@DavZim thanks for clean report, appreciate! Please try latest commit from the dev branch and see if it works

Works perfectly for me.
Thank you for the fast response.

Is there anything I can help you with to get this to CRAN?

Sure, I'll look into it.
Quick question regarding the added tests:

  • I would put the extra tests in test-app-content-handlers.R
  • As this requires writing files, I would create tempfiles() (probably even better to use tempdir() and serve this then as a static route. This would be done in the setup.R / example-app "content-handlers".

Is that OK for you or do you have a better idea?