2015-01-08 15:38:54 -05:00
|
|
|
;;; GNU Guix --- Functional package management for GNU
|
|
|
|
;;; Copyright © 2014 David Thompson <davet@gnu.org>
|
2016-12-04 23:42:49 -05:00
|
|
|
;;; Copyright © 2015, 2016 Eric Bavier <bavier@member.fsf.org>
|
2023-05-15 16:53:20 -04:00
|
|
|
;;; Copyright © 2018, 2019, 2023 Ludovic Courtès <ludo@gnu.org>
|
2020-04-14 12:01:11 -04:00
|
|
|
;;; Copyright © 2020 Ricardo Wurmus <rekado@elephly.net>
|
2015-01-08 15:38:54 -05:00
|
|
|
;;;
|
|
|
|
;;; This file is part of GNU Guix.
|
|
|
|
;;;
|
|
|
|
;;; GNU Guix is free software; you can redistribute it and/or modify it
|
|
|
|
;;; under the terms of the GNU General Public License as published by
|
|
|
|
;;; the Free Software Foundation; either version 3 of the License, or (at
|
|
|
|
;;; your option) any later version.
|
|
|
|
;;;
|
|
|
|
;;; GNU Guix is distributed in the hope that it will be useful, but
|
|
|
|
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
;;; GNU General Public License for more details.
|
|
|
|
;;;
|
|
|
|
;;; You should have received a copy of the GNU General Public License
|
|
|
|
;;; along with GNU Guix. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
(define-module (guix import json)
|
|
|
|
#:use-module (json)
|
2016-12-04 23:42:49 -05:00
|
|
|
#:use-module (guix http-client)
|
2015-01-08 15:38:54 -05:00
|
|
|
#:use-module (guix import utils)
|
2020-04-14 12:01:11 -04:00
|
|
|
#:use-module (guix import print)
|
2020-04-14 18:43:39 -04:00
|
|
|
#:use-module (ice-9 match)
|
2020-04-14 12:01:11 -04:00
|
|
|
#:use-module (ice-9 rdelim)
|
2020-04-14 18:43:39 -04:00
|
|
|
#:use-module (srfi srfi-1)
|
2020-04-14 12:01:11 -04:00
|
|
|
#:use-module (srfi srfi-2)
|
2020-04-14 18:43:39 -04:00
|
|
|
#:use-module (srfi srfi-26)
|
2016-12-04 23:42:49 -05:00
|
|
|
#:use-module (srfi srfi-34)
|
2020-04-14 12:01:11 -04:00
|
|
|
#:export (json-fetch
|
2020-04-14 18:38:15 -04:00
|
|
|
json->code
|
2020-04-14 12:01:11 -04:00
|
|
|
json->scheme-file))
|
2015-01-08 15:38:54 -05:00
|
|
|
|
2018-08-20 09:11:14 -04:00
|
|
|
(define* (json-fetch url
|
2022-05-18 14:10:55 -04:00
|
|
|
#:key
|
|
|
|
(http-fetch http-fetch)
|
2023-05-15 16:53:20 -04:00
|
|
|
(timeout 10)
|
2018-08-20 09:11:14 -04:00
|
|
|
;; Note: many websites returns 403 if we omit a
|
|
|
|
;; 'User-Agent' header.
|
2022-05-18 14:10:55 -04:00
|
|
|
(headers `((user-agent . "GNU Guile")
|
|
|
|
(Accept . "application/json"))))
|
2018-06-10 14:35:39 -04:00
|
|
|
"Return a representation of the JSON resource URL (a list or hash table), or
|
2018-08-20 09:11:14 -04:00
|
|
|
#f if URL returns 403 or 404. HEADERS is a list of HTTP headers to pass in
|
2022-05-18 14:10:55 -04:00
|
|
|
the query. HTTP-FETCH is called to perform the request: for example, to
|
|
|
|
enable caching, supply 'http-fetch/cached'."
|
2016-12-04 23:42:49 -05:00
|
|
|
(guard (c ((and (http-get-error? c)
|
2018-06-10 14:35:39 -04:00
|
|
|
(let ((error (http-get-error-code c)))
|
|
|
|
(or (= 403 error)
|
|
|
|
(= 404 error))))
|
|
|
|
#f))
|
2023-05-15 16:53:20 -04:00
|
|
|
(let* ((port (http-fetch url #:timeout timeout #:headers headers))
|
2018-06-10 14:35:39 -04:00
|
|
|
(result (json->scm port)))
|
2016-12-04 23:42:49 -05:00
|
|
|
(close-port port)
|
|
|
|
result)))
|
2020-04-14 12:01:11 -04:00
|
|
|
|
|
|
|
(define (json->code file-name)
|
2020-04-14 18:43:39 -04:00
|
|
|
"Read FILE-NAME containing one ore more JSON package definitions and return
|
|
|
|
a list of S-expressions, or return #F when the JSON is invalid."
|
2020-04-14 12:01:11 -04:00
|
|
|
(catch 'json-invalid
|
|
|
|
(lambda ()
|
|
|
|
(let ((json (json-string->scm
|
|
|
|
(with-input-from-file file-name read-string))))
|
2020-04-14 18:43:39 -04:00
|
|
|
(match json
|
|
|
|
(#(packages ...)
|
|
|
|
;; To allow definitions to refer to one another, collect references
|
|
|
|
;; to local definitions and tell alist->package to ignore them.
|
|
|
|
(second
|
|
|
|
(memq #:result
|
|
|
|
(fold
|
|
|
|
(lambda (pkg names+result)
|
|
|
|
(match names+result
|
|
|
|
((#:names names #:result result)
|
|
|
|
(list #:names
|
|
|
|
(cons (assoc-ref pkg "name") names)
|
|
|
|
#:result
|
|
|
|
(append result
|
|
|
|
(list
|
|
|
|
(package->code (alist->package pkg names))
|
|
|
|
(string->symbol (assoc-ref pkg "name"))))))))
|
|
|
|
(list #:names '()
|
|
|
|
#:result '())
|
|
|
|
packages))))
|
|
|
|
(package
|
|
|
|
(list (package->code (alist->package json))
|
|
|
|
(string->symbol (assoc-ref json "name")))))))
|
2020-04-14 12:01:11 -04:00
|
|
|
(const #f)))
|
|
|
|
|
|
|
|
(define (json->scheme-file file)
|
|
|
|
"Convert the FILE containing a JSON package definition to a Scheme
|
|
|
|
representation and return the new file name (or #F on error)."
|
2020-04-14 18:43:39 -04:00
|
|
|
(and-let* ((sexprs (json->code file))
|
2020-04-14 12:01:11 -04:00
|
|
|
(file* (let* ((tempdir (or (getenv "TMPDIR") "/tmp"))
|
|
|
|
(template (string-append tempdir "/guix-XXXXXX"))
|
|
|
|
(port (mkstemp! template)))
|
|
|
|
(close-port port)
|
|
|
|
template)))
|
|
|
|
(call-with-output-file file*
|
|
|
|
(lambda (port)
|
|
|
|
(write '(use-modules (gnu)
|
|
|
|
(guix)
|
|
|
|
((guix licenses) #:prefix license:))
|
|
|
|
port)
|
2020-04-14 18:43:39 -04:00
|
|
|
(for-each (cut write <> port) sexprs)))
|
2020-04-14 12:01:11 -04:00
|
|
|
file*))
|