Cadence Ember
cf74ffb0e2
* Rewrite archiver.rkt to manage the stages order * Rewrite archiver-gui.rkt: * Remembers the previous incomplete queue items * Pretty graphics for icons and progress bars * Segmented progress bars to indicate different stages * Fix archiver-cli.rkt to use new stages * Switch to req -d, so it doesn't auto-install gui libs
335 lines
16 KiB
Racket
335 lines
16 KiB
Racket
#lang racket/base
|
|
(require racket/file
|
|
racket/function
|
|
racket/list
|
|
racket/path
|
|
racket/sequence
|
|
racket/string
|
|
net/url
|
|
net/mime
|
|
file/sha1
|
|
net/http-easy
|
|
db
|
|
json
|
|
"archiver-database.rkt"
|
|
"../lib/html-parsing/main.rkt"
|
|
"../lib/mime-types.rkt"
|
|
"../lib/syntax.rkt"
|
|
"../lib/tree-updater.rkt"
|
|
"../lib/url-utils.rkt"
|
|
"../lib/xexpr-utils.rkt"
|
|
"../lib/archive-file-mappings.rkt")
|
|
|
|
(provide
|
|
basename->name-for-query
|
|
image-url->values
|
|
hash->save-dir
|
|
all-stages)
|
|
|
|
(module+ test
|
|
(require rackunit))
|
|
|
|
(define archive-root (anytime-path ".." "storage/archive"))
|
|
(make-directory* archive-root)
|
|
|
|
(define sources '#hasheq((style . 1) (page . 2)))
|
|
|
|
(define (get-origin wikiname)
|
|
(format "https://~a.fandom.com" wikiname))
|
|
|
|
(define (insert-wiki-entry wikiname)
|
|
(define dest-url
|
|
(format "https://~a.fandom.com/api.php?~a"
|
|
wikiname
|
|
(params->query '(("action" . "query")
|
|
("meta" . "siteinfo")
|
|
("siprop" . "general|rightsinfo|statistics")
|
|
("format" . "json")
|
|
("formatversion" . "2")))))
|
|
(define data (response-json (get dest-url)))
|
|
(define exists? (query-maybe-value* "select progress from wiki where wikiname = ?" wikiname))
|
|
(if (and exists? (not (sql-null? exists?)))
|
|
(query-exec* "update wiki set sitename = ?, basepage = ?, license_text = ?, license_url = ? where wikiname = ?"
|
|
(jp "/query/general/sitename" data)
|
|
(second (regexp-match #rx"/wiki/(.*)" (jp "/query/general/base" data)))
|
|
(jp "/query/rightsinfo/text" data)
|
|
(jp "/query/rightsinfo/url" data)
|
|
wikiname)
|
|
(query-exec* "insert into wiki (wikiname, progress, sitename, basepage, license_text, license_url) values (?, 0, ?, ?, ?, ?)"
|
|
wikiname
|
|
(jp "/query/general/sitename" data)
|
|
(second (regexp-match #rx"/wiki/(.*)" (jp "/query/general/base" data)))
|
|
(jp "/query/rightsinfo/text" data)
|
|
(jp "/query/rightsinfo/url" data)))
|
|
(jp "/query/statistics/articles" data))
|
|
|
|
|
|
(define (check-style-for-images wikiname path)
|
|
(define content (file->string path))
|
|
(define urls (regexp-match* #rx"url\\(\"?'?([^)]*)'?\"?\\)" content #:match-select cadr))
|
|
(for/list ([url urls]
|
|
#:when (not (or (equal? url "")
|
|
(equal? url "'")
|
|
(string-suffix? url "\"")
|
|
(string-contains? url "/resources-ucp/")
|
|
(string-contains? url "/fonts/")
|
|
(string-contains? url "/drm_fonts/")
|
|
(string-contains? url "//db.onlinewebfonts.com/")
|
|
(string-contains? url "//bits.wikimedia.org/")
|
|
(string-contains? url "dropbox")
|
|
(string-contains? url "only=styles")
|
|
(string-contains? url "https://https://")
|
|
(regexp-match? #rx"^%20" url)
|
|
(regexp-match? #rx"^data:" url))))
|
|
(cond
|
|
[(string-prefix? url "https://") url]
|
|
[(string-prefix? url "http://") (regexp-replace #rx"http:" url "https:")]
|
|
[(string-prefix? url "//") (string-append "https:" url)]
|
|
[(string-prefix? url "/") (format "https://~a.fandom.com~a" wikiname url)]
|
|
[else (raise-user-error "While calling check-style-for-images, this URL had an unknown format and couldn't be saved:" url path)])))
|
|
|
|
(define (download-styles-for-wiki wikiname callback)
|
|
(define save-dir (build-path archive-root wikiname "styles"))
|
|
(make-directory* save-dir)
|
|
(define theme (λ (theme-name)
|
|
(cons (format "https://~a.fandom.com/wikia.php?controller=ThemeApi&method=themeVariables&variant=~a" wikiname theme-name)
|
|
(build-path save-dir (format "themeVariables-~a.css" theme-name)))))
|
|
;; (Listof (Pair url save-path))
|
|
(define styles
|
|
(list
|
|
(theme "default")
|
|
(theme "light")
|
|
(theme "dark")
|
|
(cons (format "https://~a.fandom.com/load.php?lang=en&modules=skin.fandomdesktop.styles%7Cext.fandom.PortableInfoboxFandomDesktop.css%7Cext.fandom.GlobalComponents.CommunityHeaderBackground.css%7Cext.gadget.site-styles%2Csound-styles%7Csite.styles&only=styles&skin=fandomdesktop" wikiname)
|
|
(build-path save-dir "site.css"))))
|
|
(for ([style styles]
|
|
[i (in-naturals)])
|
|
(callback i (length styles) "styles...")
|
|
(define r (get (car style)))
|
|
(define body (response-body r))
|
|
(display-to-file body (cdr style) #:exists 'replace)
|
|
;; XXX: how the HELL do I deal with @import?? would need some kind of recursion here. how will the page server know where to look up the style file to be able to serve them again? do I add another link-stylesheet tag to the main page? what about the remaining stuck @import url?
|
|
)
|
|
(callback (length styles) (length styles) "styles...")
|
|
styles)
|
|
|
|
(define (hash->save-dir wikiname hash)
|
|
(build-path archive-root wikiname "images" (substring hash 0 1) (substring hash 0 2)))
|
|
|
|
(define (image-url->values i)
|
|
;; TODO: handle case where there is multiple broken cb parameter on minecraft wiki
|
|
;; TODO: ensure it still "works" with broken & on minecraft wiki
|
|
(define no-cb (regexp-replace #rx"\\cb=[0-9]+&?" i "")) ; remove cb url parameter which does nothing
|
|
(define key (regexp-replace #rx"[&?]$" no-cb "")) ; remove extra separator if necessary
|
|
(define hash (sha1 (string->bytes/utf-8 key)))
|
|
(cons key hash))
|
|
|
|
|
|
;; 1. Download list of wiki pages and store in database, if not done yet for that wiki
|
|
(define (if-necessary-download-list-of-pages wikiname callback)
|
|
(define wiki-progress (query-maybe-value* "select progress from wiki where wikiname = ?" wikiname))
|
|
;; done yet?
|
|
(unless (and (real? wiki-progress) (wiki-progress . >= . 1))
|
|
;; Count total pages
|
|
(define num-pages (insert-wiki-entry wikiname))
|
|
;; Download the entire index of pages
|
|
(define basenames
|
|
(let loop ([path-with-namefrom "/wiki/Local_Sitemap"]
|
|
[basenames-previous-pages null])
|
|
;; Download the current index page
|
|
(define url (format "https://~a.fandom.com~a" wikiname path-with-namefrom))
|
|
(define r (get url))
|
|
;; Metadata from this page (the link to the next page)
|
|
(define page (html->xexp (bytes->string/utf-8 (response-body r))))
|
|
(define link-namefrom
|
|
((query-selector (λ (t a c x) (and (eq? t 'a)
|
|
(pair? x)
|
|
(string-contains? (car x) "Next page")
|
|
(let ([href (get-attribute 'href a)] )
|
|
(and href (string-contains? href "/wiki/Local_Sitemap")))))
|
|
page #:include-text? #t)))
|
|
;; Content from this page
|
|
(define basenames-this-page
|
|
(for/list ([link (in-producer
|
|
(query-selector
|
|
(λ (t a c) (eq? t 'a))
|
|
((query-selector (λ (t a c) (has-class? "mw-allpages-chunk" a)) page)))
|
|
#f)])
|
|
(local-encoded-url->basename (get-attribute 'href (bits->attributes link)))))
|
|
;; Call the progress callback
|
|
(define all-basenames (append basenames-previous-pages basenames-this-page))
|
|
(callback (length all-basenames) num-pages (last all-basenames))
|
|
;; Recurse to download from the next page
|
|
(if link-namefrom
|
|
(loop (get-attribute 'href (bits->attributes link-namefrom)) all-basenames)
|
|
all-basenames)))
|
|
;; Save those pages into the database
|
|
;; SQLite can have a maximum of 32766 parameters in a single query
|
|
(for ([slice (in-slice 32760 basenames)])
|
|
(define query-template (string-join (make-list (length slice) "(?1, ?, 0)") ", " #:before-first "insert or ignore into page (wikiname, basename, progress) values "))
|
|
(call-with-transaction
|
|
(get-slc)
|
|
(λ ()
|
|
(apply query-exec* query-template wikiname slice)
|
|
;; Record that we have the complete list of pages
|
|
(query-exec* "update wiki set progress = 1 where wikiname = ?" wikiname))))))
|
|
|
|
|
|
;; 2. Download each page via API and:
|
|
;; * Save API response to file
|
|
(define max-page-progress 1)
|
|
(define (save-each-page wikiname callback)
|
|
;; prepare destination folder
|
|
(define save-dir (build-path archive-root wikiname))
|
|
(make-directory* save-dir)
|
|
;; gather list of basenames to download (that aren't yet complete)
|
|
(define basenames (query-list* "select basename from page where wikiname = ? and progress < ?"
|
|
wikiname max-page-progress))
|
|
;; counter of complete/incomplete basenames
|
|
(define already-done-count
|
|
(query-value* "select count(*) from page where wikiname = ? and progress = ?"
|
|
wikiname max-page-progress))
|
|
(define not-done-count
|
|
(query-value* "select count(*) from page where wikiname = ? and progress < ?"
|
|
wikiname max-page-progress))
|
|
(define total-count (+ already-done-count not-done-count))
|
|
;; set initial progress
|
|
(callback already-done-count total-count "")
|
|
;; loop through basenames and download
|
|
(for ([basename basenames]
|
|
[i (in-naturals (add1 already-done-count))])
|
|
(define name-for-query (basename->name-for-query basename))
|
|
(define dest-url
|
|
(format "https://~a.fandom.com/api.php?~a"
|
|
wikiname
|
|
(params->query `(("action" . "parse")
|
|
("page" . ,name-for-query)
|
|
("prop" . "text|headhtml|langlinks")
|
|
("formatversion" . "2")
|
|
("format" . "json")))))
|
|
(define r (get dest-url))
|
|
(define body (response-body r))
|
|
(define filename (string-append basename ".json"))
|
|
(define save-path
|
|
(cond [((string-length basename) . > . 240)
|
|
(define key (sha1 (string->bytes/latin-1 basename)))
|
|
(query-exec* "insert into special_page (wikiname, key, basename) values (?, ?, ?)"
|
|
wikiname key basename)
|
|
(build-path save-dir (string-append key ".json"))]
|
|
[#t
|
|
(build-path save-dir (string-append basename ".json"))]))
|
|
(display-to-file body save-path #:exists 'replace)
|
|
(query-exec* "update page set progress = 1 where wikiname = ? and basename = ?"
|
|
wikiname basename)
|
|
(callback i total-count basename))
|
|
;; saved all pages, register that fact in the database
|
|
(query-exec* "update wiki set progress = 2 where wikiname = ?" wikiname))
|
|
|
|
|
|
;; 3. Download CSS and:
|
|
;; * Save CSS to file
|
|
;; * Record style images to database
|
|
(define (if-necessary-download-and-check-styles wikiname callback)
|
|
(define wiki-progress (query-maybe-value* "select progress from wiki where wikiname = ?" wikiname))
|
|
(unless (and (number? wiki-progress) (wiki-progress . >= . 3))
|
|
(define styles (download-styles-for-wiki wikiname callback))
|
|
(define unique-image-urls
|
|
(remove-duplicates
|
|
(map image-url->values
|
|
(flatten
|
|
(for/list ([style styles])
|
|
(check-style-for-images wikiname (cdr style)))))
|
|
#:key cdr))
|
|
(for ([pair unique-image-urls])
|
|
(query-exec* "insert or ignore into image (wikiname, url, hash, ext, source, progress) values (?, ?, ?, NULL, 1, 0)" wikiname (car pair) (cdr pair)))
|
|
(query-exec* "update wiki set progress = 3 where wikiname = ?" wikiname)))
|
|
|
|
|
|
;; 4: From downloaded pages, record URLs of image sources and inline style images to database
|
|
(define (check-json-for-images wikiname path)
|
|
(define data (with-input-from-file path (λ () (read-json))))
|
|
(define page (html->xexp (preprocess-html-wiki (jp "/parse/text" data))))
|
|
(define tree (update-tree-wiki page wikiname))
|
|
null
|
|
#;(remove-duplicates
|
|
(for/list ([element (in-producer
|
|
(query-selector
|
|
(λ (t a c)
|
|
(and (eq? t 'img)
|
|
(get-attribute 'src a)))
|
|
tree)
|
|
#f)])
|
|
(image-url->values (get-attribute 'src (bits->attributes element))))))
|
|
|
|
|
|
;; 5. Download image sources and style images according to database
|
|
(define (save-each-image wikiname callback)
|
|
(define source (hash-ref sources 'style)) ;; TODO: download entire wiki images instead?
|
|
;; gather list of basenames to download (that aren't yet complete)
|
|
(define rows (query-rows* "select url, hash from image where wikiname = ? and source <= ? and progress < 1"
|
|
wikiname source))
|
|
;; counter of complete/incomplete basenames
|
|
(define already-done-count
|
|
(query-value* "select count(*) from image where wikiname = ? and source <= ? and progress = 1"
|
|
wikiname source))
|
|
(define not-done-count
|
|
(query-value* "select count(*) from image where wikiname = ? and source <= ? and progress < 1"
|
|
wikiname source))
|
|
;; set initial progress
|
|
(callback already-done-count (+ already-done-count not-done-count) "")
|
|
;; loop through urls and download
|
|
(for ([row rows]
|
|
[i (in-naturals 1)])
|
|
;; row fragments
|
|
(define url (vector-ref row 0))
|
|
(define hash (vector-ref row 1))
|
|
;; check
|
|
#; (printf "~a -> ~a~n" url hash)
|
|
(define r (get url))
|
|
(define declared-type (response-headers-ref r 'content-type))
|
|
(define final-type (if (equal? declared-type #"application/octet-stream")
|
|
(let ([sniff-entity (message-entity (mime-analyze (response-body r)))])
|
|
(string->bytes/latin-1 (format "~a/~a" (entity-type sniff-entity) (entity-subtype sniff-entity))))
|
|
declared-type))
|
|
(define ext
|
|
(with-handlers ([exn:fail:contract? (λ _ (error 'save-each-image "no ext found for mime type `~a` in file ~a" final-type url))])
|
|
(bytes->string/latin-1 (mime-type->ext final-type))))
|
|
;; save
|
|
(define save-dir (hash->save-dir wikiname hash))
|
|
(make-directory* save-dir)
|
|
(define save-path (build-path save-dir (string-append hash "." ext)))
|
|
(define body (response-body r))
|
|
(display-to-file body save-path #:exists 'replace)
|
|
(query-exec* "update image set progress = 1, ext = ? where wikiname = ? and hash = ?"
|
|
ext wikiname hash)
|
|
(callback (+ already-done-count i) (+ already-done-count not-done-count) (string-append (substring hash 0 6) "..." ext)))
|
|
;; saved all images, register that fact in the database
|
|
(query-exec* "update wiki set progress = 4 where wikiname = ?" wikiname))
|
|
|
|
(define all-stages
|
|
(list
|
|
if-necessary-download-list-of-pages
|
|
save-each-page
|
|
if-necessary-download-and-check-styles
|
|
;; check-json-for-images
|
|
save-each-image))
|
|
|
|
(module+ test
|
|
(check-equal? (html->xexp "<img src=\"https://example.com/images?src=Blah.jpg&width=150\">")
|
|
'(*TOP* (img (@ (src "https://example.com/images?src=Blah.jpg&width=150")))))
|
|
#;(download-list-of-pages "minecraft" values)
|
|
#;(save-each-page "minecraft" values)
|
|
#;(check-json-for-images "chiki" (build-path archive-root "chiki" "Fiona.json"))
|
|
#;(do-step-3 "gallowmere")
|
|
#;(save-each-image "gallowmere" (hash-ref sources 'style) (λ (a b c) (printf "~a/~a ~a~n" a b c)))
|
|
|
|
#;(for ([wikiname (query-list* "select wikiname from wiki")])
|
|
(println wikiname)
|
|
(insert-wiki-entry wikiname))
|
|
|
|
#;(for ([wikiname (query-list* "select wikiname from wiki")])
|
|
(println wikiname)
|
|
(do-step-3 wikiname)
|
|
(save-each-image wikiname (hash-ref sources 'style) (λ (a b c) (printf "~a/~a ~a~n" a b c)))))
|
|
|
|
; (for ([stage all-stages]) (stage "create" (λ (a b c) (printf "~a/~a ~a~n" a b c))))
|