110 lines
4.1 KiB
Racket
110 lines
4.1 KiB
Racket
#lang racket/base
|
|
(require racket/dict
|
|
racket/function
|
|
racket/list
|
|
racket/match
|
|
racket/string
|
|
; libs
|
|
(prefix-in easy: net/http-easy)
|
|
; html libs
|
|
"../lib/html-parsing/main.rkt"
|
|
html-writing
|
|
; web server libs
|
|
net/url
|
|
web-server/http
|
|
web-server/dispatchers/dispatch
|
|
; my libs
|
|
"application-globals.rkt"
|
|
"config.rkt"
|
|
"data.rkt"
|
|
"fandom-request.rkt"
|
|
"../lib/pure-utils.rkt"
|
|
"../lib/syntax.rkt"
|
|
"../lib/thread-utils.rkt"
|
|
"../lib/tree-updater.rkt"
|
|
"../lib/url-utils.rkt"
|
|
"../lib/xexpr-utils.rkt")
|
|
|
|
(provide
|
|
; used by the web server
|
|
page-wiki
|
|
; used by page-category, and similar pages that are partially wiki pages
|
|
update-tree-wiki
|
|
preprocess-html-wiki)
|
|
|
|
(module+ test
|
|
(require rackunit))
|
|
|
|
(define (page-wiki req)
|
|
(define wikiname (path/param-path (first (url-path (request-uri req)))))
|
|
(define user-cookies (user-cookies-getter req))
|
|
(define path (string-join (map path/param-path (cddr (url-path (request-uri req)))) "/"))
|
|
(define source-url (format "https://~a.fandom.com/wiki/~a" wikiname path))
|
|
|
|
(define-values (dest-res siteinfo)
|
|
(thread-values
|
|
(λ ()
|
|
(fandom-get-api
|
|
wikiname
|
|
`(("action" . "parse")
|
|
("page" . ,path)
|
|
("prop" . "text|headhtml|langlinks")
|
|
("formatversion" . "2")
|
|
("format" . "json"))
|
|
#:headers `#hasheq((cookie . ,(format "theme=~a" (user-cookies^-theme user-cookies))))))
|
|
(λ ()
|
|
(siteinfo-fetch wikiname))))
|
|
|
|
(cond
|
|
[(eq? 200 (easy:response-status-code dest-res))
|
|
(let* ([data (easy:response-json dest-res)]
|
|
[title (jp "/parse/title" data "")]
|
|
[page-html (jp "/parse/text" data "")]
|
|
[page-html (preprocess-html-wiki page-html)]
|
|
[page (html->xexp page-html)]
|
|
[head-data ((head-data-getter wikiname) data)])
|
|
(if (equal? "missingtitle" (jp "/error/code" data #f))
|
|
(next-dispatcher)
|
|
(response-handler
|
|
(define body
|
|
(generate-wiki-page
|
|
(update-tree-wiki page wikiname)
|
|
#:req req
|
|
#:source-url source-url
|
|
#:wikiname wikiname
|
|
#:title title
|
|
#:head-data head-data
|
|
#:siteinfo siteinfo))
|
|
(define redirect-query-parameter (dict-ref (url-query (request-uri req)) 'redirect "yes"))
|
|
(define redirect-msg ((query-selector (attribute-selector 'class "redirectMsg") body)))
|
|
(define redirect-msg-a (if redirect-msg
|
|
((query-selector (λ (t a c) (eq? t 'a)) redirect-msg))
|
|
#f))
|
|
(define headers
|
|
(build-headers
|
|
always-headers
|
|
; redirect-query-parameter: only the string "no" is significant:
|
|
; https://github.com/Wikia/app/blob/fe60579a53f16816d65dad1644363160a63206a6/includes/Wiki.php#L367
|
|
(when (and redirect-msg-a
|
|
(not (equal? redirect-query-parameter "no")))
|
|
(let* ([dest (get-attribute 'href (bits->attributes redirect-msg-a))]
|
|
[value (bytes-append #"0;url=" (string->bytes/utf-8 dest))])
|
|
(header #"Refresh" value)))))
|
|
(when (config-true? 'debug)
|
|
; used for its side effects
|
|
; convert to string with error checking, error will be raised if xexp is invalid
|
|
(xexp->html body))
|
|
(response/output
|
|
#:code 200
|
|
#:headers headers
|
|
(λ (out)
|
|
(write-html body out))))))]
|
|
[(eq? 404 (easy:response-status-code dest-res))
|
|
(next-dispatcher)]
|
|
[else
|
|
(response-handler
|
|
(error 'page-wiki "Tried to load page ~a/~v~nSadly, the page didn't load because Fandom returned status code ~a with response:~n~a"
|
|
wikiname
|
|
path
|
|
(easy:response-status-code dest-res)
|
|
(easy:response-body dest-res)))]))
|