|
| 1 | +(in-package :cl-user) |
| 2 | +(defpackage quickdocs-updater.release |
| 3 | + (:use :cl |
| 4 | + :split-sequence) |
| 5 | + (:import-from :quickdocs-updater.extracter |
| 6 | + :extractedp |
| 7 | + :extract-result-directory |
| 8 | + :release-systems-info) |
| 9 | + (:import-from :alexandria |
| 10 | + :when-let |
| 11 | + :starts-with-subseq |
| 12 | + :compose |
| 13 | + :ignore-some-conditions) |
| 14 | + (:export :readme-file |
| 15 | + :release-version |
| 16 | + :primary-system |
| 17 | + :project-homepage) |
| 18 | + (:documentation "Function/method collections for Quicklisp releases.")) |
| 19 | +(in-package :quickdocs-updater.release) |
| 20 | + |
| 21 | +(defun readme-file (release) |
| 22 | + (check-type release ql-dist:release) |
| 23 | + (assert (ql-dist:installedp release)) |
| 24 | + (find "README" |
| 25 | + (uiop:directory-files (ql-dist:base-directory release)) |
| 26 | + :key #'pathname-name |
| 27 | + :test #'string=)) |
| 28 | + |
| 29 | +(defun release-version (release) |
| 30 | + "Return the latest Quicklisp dist version that `release` was updated." |
| 31 | + (check-type release ql-dist:release) |
| 32 | + (when-let (match (nth-value 1 |
| 33 | + (ppcre:scan-to-strings "beta\\.quicklisp\\.org/archive/[^/]+/([^/]+)" |
| 34 | + (ql-dist:archive-url release)))) |
| 35 | + (aref match 0))) |
| 36 | + |
| 37 | +(defun primary-system (release) |
| 38 | + (check-type release ql-dist:release) |
| 39 | + (flet ((remove-cl-prefix (name) |
| 40 | + (if (starts-with-subseq "cl-" name) |
| 41 | + (subseq name 3) |
| 42 | + name))) |
| 43 | + (let ((project-name (remove-cl-prefix (ql-dist:project-name release))) |
| 44 | + (provided-systems (ql-dist:provided-systems release))) |
| 45 | + (or (find project-name |
| 46 | + provided-systems |
| 47 | + :key (compose #'remove-cl-prefix #'ql-dist:name) |
| 48 | + :test #'string=) |
| 49 | + (first provided-systems))))) |
| 50 | + |
| 51 | +;; NOTE: quicklisp-projects is updated every month and it could be different from the `release' has been included. |
| 52 | +;; It would be a problem when updating old dists. |
| 53 | +(defun project-source-txt (release) |
| 54 | + (check-type release ql-dist:release) |
| 55 | + (merge-pathnames (format nil "~A/source.txt" (ql-dist:name release)) |
| 56 | + (asdf:system-relative-pathname :quickdocs-database |
| 57 | + #P"modules/quicklisp-projects/"))) |
| 58 | + |
| 59 | +(defun project-source-info (release) |
| 60 | + (check-type release ql-dist:release) |
| 61 | + (let ((data (uiop:read-file-string (project-source-txt release)))) |
| 62 | + (destructuring-bind (type source-url) |
| 63 | + (split-sequence #\Space data :count 2) |
| 64 | + (values type source-url)))) |
| 65 | + |
| 66 | +(defun project-homepage (release) |
| 67 | + (check-type release ql-dist:release) |
| 68 | + (let ((systems-info (release-systems-info release))) |
| 69 | + (or |
| 70 | + ;; Use if :homepage exists in ASD files |
| 71 | + (getf (find-if (lambda (info) |
| 72 | + (getf info :homepage)) |
| 73 | + systems-info) |
| 74 | + :homepage) |
| 75 | + |
| 76 | + ;; Use the repository URL if it's on some popular services, like GitHub or BitBucket |
| 77 | + (multiple-value-bind (type source-url) |
| 78 | + (project-source-info release) |
| 79 | + (ignore-some-conditions (quri:uri-error) |
| 80 | + (let* ((uri (quri:uri source-url)) |
| 81 | + (domain (quri:uri-domain uri))) |
| 82 | + (cond |
| 83 | + ((string= domain "github.com") |
| 84 | + (let ((repos-id (ppcre:regex-replace "\\.[^\\.]*$" (quri:uri-path uri) ""))) |
| 85 | + ;; TODO: request website URL to GitHub |
| 86 | + (concatenate 'string |
| 87 | + "https://github.com" |
| 88 | + repos-id))) |
| 89 | + ((string= domain "bitbucket.org") |
| 90 | + ;; TODO: request website URL to BitBucket |
| 91 | + source-url) |
| 92 | + ((string= domain "gitlab.common-lisp.net") |
| 93 | + (let ((repos-id (ppcre:regex-replace "\\.[^\\.]*$" (quri:uri-path uri) ""))) |
| 94 | + (concatenate 'string |
| 95 | + "http://gitlab.common-lisp.net" |
| 96 | + repos-id))) |
| 97 | + ((string= domain "common-lisp.net") |
| 98 | + (let ((match (ppcre:scan-to-strings "://common-lisp\\.net/project/([^\\/]+)" source-url))) |
| 99 | + (format nil "http://common-lisp.net/project/~A" |
| 100 | + (quri:url-encode |
| 101 | + (if match |
| 102 | + (aref match 0) |
| 103 | + (ql-dist:project-name release)) |
| 104 | + :encoding :utf-8)))) |
| 105 | + ((or (string= domain "weitz.de") |
| 106 | + (string= type "ediware-http")) |
| 107 | + (format nil "http://weitz.de/~A/" |
| 108 | + (quri:url-encode (ql-dist:project-name release) :encoding :utf-8)))))))))) |
0 commit comments