2019-02-06 16:14:44 +00:00
|
|
|
;;; Guix Data Service -- Information about Guix over time
|
|
|
|
|
;;; Copyright © 2017 Ricardo Wurmus <rekado@elephly.net>
|
2023-01-01 12:27:34 +00:00
|
|
|
;;; Copyright © 2019, 2020, 2022, 2023 Christopher Baines <mail@cbaines.net>
|
2019-02-06 16:14:44 +00:00
|
|
|
;;;
|
|
|
|
|
;;; This program is free software: you can redistribute it and/or
|
|
|
|
|
;;; modify it under the terms of the GNU Affero General Public License
|
|
|
|
|
;;; as published by the Free Software Foundation, either version 3 of
|
|
|
|
|
;;; the License, or (at your option) any later version.
|
|
|
|
|
;;;
|
|
|
|
|
;;; This program 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
|
|
|
|
|
;;; Affero General Public License for more details.
|
|
|
|
|
;;;
|
|
|
|
|
;;; You should have received a copy of the GNU Affero General Public
|
|
|
|
|
;;; License along with this program. If not, see
|
|
|
|
|
;;; <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
(define-module (guix-data-service web server)
|
|
|
|
|
#:use-module (srfi srfi-1)
|
2019-05-11 22:56:25 +01:00
|
|
|
#:use-module (srfi srfi-11)
|
2025-01-29 16:31:50 +00:00
|
|
|
#:use-module (srfi srfi-71)
|
2020-03-14 13:13:48 +00:00
|
|
|
#:use-module (ice-9 match)
|
2024-07-16 21:50:36 +01:00
|
|
|
#:use-module (ice-9 threads)
|
2019-02-06 16:14:44 +00:00
|
|
|
#:use-module (web http)
|
|
|
|
|
#:use-module (web request)
|
|
|
|
|
#:use-module (web uri)
|
2020-03-14 13:13:48 +00:00
|
|
|
#:use-module (system repl error-handling)
|
2022-06-17 12:55:05 +01:00
|
|
|
#:use-module (ice-9 atomic)
|
2023-07-09 16:52:35 +01:00
|
|
|
#:use-module (fibers)
|
2024-07-16 21:50:36 +01:00
|
|
|
#:use-module (fibers channels)
|
2023-11-16 11:30:23 +00:00
|
|
|
#:use-module (fibers scheduler)
|
2023-07-09 16:52:35 +01:00
|
|
|
#:use-module (fibers conditions)
|
2025-03-10 21:45:16 +00:00
|
|
|
#:use-module (knots)
|
2024-12-15 19:08:28 +00:00
|
|
|
#:use-module (knots web-server)
|
2025-03-10 21:48:02 +00:00
|
|
|
#:use-module (knots thread-pool)
|
2024-12-15 19:08:28 +00:00
|
|
|
#:use-module (knots resource-pool)
|
2023-01-01 12:27:34 +00:00
|
|
|
#:use-module (prometheus)
|
2023-07-09 16:52:35 +01:00
|
|
|
#:use-module (guix-data-service utils)
|
2023-01-01 12:43:19 +00:00
|
|
|
#:use-module (guix-data-service database)
|
2019-02-06 16:14:44 +00:00
|
|
|
#:use-module (guix-data-service web controller)
|
|
|
|
|
#:use-module (guix-data-service web util)
|
2025-01-29 16:31:50 +00:00
|
|
|
#:use-module (guix-data-service web render)
|
|
|
|
|
#:use-module (guix-data-service web view html)
|
2025-05-14 15:30:07 +01:00
|
|
|
#:use-module (guix-data-service model build-status)
|
|
|
|
|
#:use-module (guix-data-service model blocked-builds)
|
2024-04-01 21:51:29 +01:00
|
|
|
#:use-module (guix-data-service model guix-revision-package-derivation)
|
2025-05-14 15:30:07 +01:00
|
|
|
#:use-module (guix-data-service model build-background-processing-queue)
|
2023-11-24 16:11:42 +00:00
|
|
|
#:export (%guix-data-service-metrics-registry
|
|
|
|
|
|
|
|
|
|
start-guix-data-service-web-server))
|
2019-02-06 16:14:44 +00:00
|
|
|
|
2022-06-17 12:55:05 +01:00
|
|
|
(define (check-startup-completed startup-completed)
|
|
|
|
|
(if (atomic-box-ref startup-completed)
|
|
|
|
|
(begin
|
|
|
|
|
;; Just in case this atomic-box-ref is expensive, only do it when
|
|
|
|
|
;; necessary
|
|
|
|
|
(set! check-startup-completed (const #t))
|
|
|
|
|
#t)
|
|
|
|
|
#f))
|
|
|
|
|
|
2024-04-02 12:04:23 +01:00
|
|
|
(define (handler request finished?
|
|
|
|
|
body controller secret-key-base startup-completed
|
2023-01-01 12:27:34 +00:00
|
|
|
render-metrics)
|
2024-04-02 12:04:23 +01:00
|
|
|
(with-exception-handler
|
|
|
|
|
(lambda (exn)
|
|
|
|
|
(with-exception-handler
|
|
|
|
|
(lambda _ #f)
|
|
|
|
|
(lambda ()
|
|
|
|
|
(simple-format (current-error-port)
|
|
|
|
|
"exception when logging: ~A\n" exn))
|
|
|
|
|
#:unwind? #t)
|
|
|
|
|
;; If we can't log, exit
|
|
|
|
|
(signal-condition! finished?))
|
|
|
|
|
(lambda ()
|
|
|
|
|
(display
|
|
|
|
|
(format #f "~a ~a\n"
|
|
|
|
|
(request-method request)
|
|
|
|
|
(uri-path (request-uri request)))))
|
|
|
|
|
#:unwind? #t)
|
2019-02-06 16:14:44 +00:00
|
|
|
(apply values
|
2020-04-24 09:00:20 +01:00
|
|
|
(let-values (((request-components mime-types)
|
|
|
|
|
(request->path-components-and-mime-type request)))
|
|
|
|
|
(controller request
|
|
|
|
|
(cons (request-method request)
|
|
|
|
|
request-components)
|
|
|
|
|
mime-types
|
|
|
|
|
body
|
2022-06-17 12:55:05 +01:00
|
|
|
secret-key-base
|
2023-01-01 12:27:34 +00:00
|
|
|
(check-startup-completed startup-completed)
|
|
|
|
|
render-metrics))))
|
2019-02-06 16:14:44 +00:00
|
|
|
|
2023-11-24 16:11:42 +00:00
|
|
|
(define %guix-data-service-metrics-registry
|
|
|
|
|
(make-parameter #f))
|
|
|
|
|
|
2022-06-17 12:55:05 +01:00
|
|
|
(define* (start-guix-data-service-web-server port host secret-key-base
|
2023-07-09 16:52:35 +01:00
|
|
|
startup-completed
|
|
|
|
|
#:key postgresql-statement-timeout
|
|
|
|
|
postgresql-connections)
|
2023-01-01 12:27:34 +00:00
|
|
|
(define registry
|
|
|
|
|
(make-metrics-registry #:namespace "guixdataservice"))
|
|
|
|
|
|
2023-01-01 12:43:19 +00:00
|
|
|
(%database-metrics-registry registry)
|
|
|
|
|
|
2023-11-24 16:11:42 +00:00
|
|
|
(%guix-data-service-metrics-registry registry)
|
|
|
|
|
|
2025-04-24 09:36:54 +01:00
|
|
|
(with-exception-handler
|
|
|
|
|
(lambda (exn)
|
|
|
|
|
(simple-format #t "failed increasing open file limit: ~A\n" exn))
|
|
|
|
|
(lambda ()
|
|
|
|
|
(setrlimit 'nofile 4096 4096))
|
|
|
|
|
#:unwind? #t)
|
|
|
|
|
|
2024-07-16 21:50:36 +01:00
|
|
|
(let ((finished? (make-condition))
|
2024-08-14 21:13:55 +01:00
|
|
|
(priority-scheduler #f)
|
2024-07-16 21:50:36 +01:00
|
|
|
(request-scheduler #f))
|
2023-07-09 16:52:35 +01:00
|
|
|
(call-with-sigint
|
|
|
|
|
(lambda ()
|
2024-08-14 21:13:55 +01:00
|
|
|
(call-with-new-thread
|
|
|
|
|
(lambda ()
|
|
|
|
|
(run-fibers
|
|
|
|
|
(lambda ()
|
|
|
|
|
(let* ((current (current-scheduler))
|
|
|
|
|
(schedulers
|
|
|
|
|
(cons current (scheduler-remote-peers current))))
|
|
|
|
|
|
|
|
|
|
(set! priority-scheduler current)
|
|
|
|
|
|
|
|
|
|
(for-each
|
|
|
|
|
(lambda (i sched)
|
|
|
|
|
(spawn-fiber
|
|
|
|
|
(lambda ()
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(set-thread-name
|
|
|
|
|
(string-append "priority " (number->string i))))
|
|
|
|
|
(const #t)))
|
|
|
|
|
sched))
|
|
|
|
|
(iota (length schedulers))
|
|
|
|
|
schedulers))
|
|
|
|
|
|
|
|
|
|
(wait finished?))
|
|
|
|
|
#:hz 0
|
|
|
|
|
#:parallelism 1)))
|
|
|
|
|
|
2023-07-09 16:52:35 +01:00
|
|
|
(run-fibers
|
|
|
|
|
(lambda ()
|
2024-07-26 21:42:21 +01:00
|
|
|
(let* ((current (current-scheduler))
|
|
|
|
|
(schedulers
|
|
|
|
|
(cons current (scheduler-remote-peers current))))
|
|
|
|
|
(for-each
|
|
|
|
|
(lambda (i sched)
|
|
|
|
|
(spawn-fiber
|
|
|
|
|
(lambda ()
|
|
|
|
|
(catch 'system-error
|
|
|
|
|
(lambda ()
|
|
|
|
|
(set-thread-name
|
|
|
|
|
(string-append "server " (number->string i))))
|
|
|
|
|
(const #t)))
|
|
|
|
|
sched))
|
|
|
|
|
(iota (length schedulers))
|
|
|
|
|
schedulers))
|
2024-07-16 21:50:36 +01:00
|
|
|
|
2024-08-14 21:13:55 +01:00
|
|
|
(while (not priority-scheduler)
|
|
|
|
|
(sleep 0.1))
|
2024-07-16 21:50:36 +01:00
|
|
|
|
|
|
|
|
(let ((requests-metric
|
|
|
|
|
(make-counter-metric registry "requests_total")))
|
|
|
|
|
|
|
|
|
|
(with-exception-handler
|
|
|
|
|
(lambda (exn)
|
|
|
|
|
(simple-format
|
|
|
|
|
(current-error-port)
|
|
|
|
|
"\n
|
2023-07-09 16:52:35 +01:00
|
|
|
error: guix-data-service could not start: ~A
|
|
|
|
|
|
|
|
|
|
Check if it's already running, or whether another process is using that
|
|
|
|
|
port. Also, the port used can be changed by passing the --port option.\n"
|
2024-07-16 21:50:36 +01:00
|
|
|
exn)
|
|
|
|
|
(primitive-exit 1))
|
|
|
|
|
(lambda ()
|
|
|
|
|
(parameterize
|
2024-09-08 10:20:21 +01:00
|
|
|
((background-connection-pool
|
|
|
|
|
(make-resource-pool
|
|
|
|
|
(lambda ()
|
|
|
|
|
(open-postgresql-connection
|
|
|
|
|
"background"
|
|
|
|
|
postgresql-statement-timeout))
|
|
|
|
|
4
|
|
|
|
|
#:name "background"
|
|
|
|
|
#:idle-seconds 5
|
|
|
|
|
#:destructor
|
|
|
|
|
(lambda (conn)
|
|
|
|
|
(close-postgresql-connection conn "background"))
|
|
|
|
|
#:scheduler priority-scheduler))
|
|
|
|
|
|
|
|
|
|
(connection-pool
|
2024-07-16 21:50:36 +01:00
|
|
|
(make-resource-pool
|
|
|
|
|
(lambda ()
|
|
|
|
|
(open-postgresql-connection
|
|
|
|
|
"web"
|
|
|
|
|
postgresql-statement-timeout))
|
|
|
|
|
(floor (/ postgresql-connections 2))
|
2024-08-14 20:12:19 +01:00
|
|
|
#:name "web"
|
2024-07-16 21:50:36 +01:00
|
|
|
#:idle-seconds 30
|
|
|
|
|
#:destructor
|
|
|
|
|
(lambda (conn)
|
2024-08-14 21:13:55 +01:00
|
|
|
(close-postgresql-connection conn "web"))
|
2025-04-27 11:40:44 +01:00
|
|
|
#:default-max-waiters 300
|
|
|
|
|
#:default-checkout-timeout (/ postgresql-statement-timeout
|
|
|
|
|
1000)
|
2024-08-14 21:13:55 +01:00
|
|
|
#:scheduler priority-scheduler))
|
2024-07-16 21:50:36 +01:00
|
|
|
|
|
|
|
|
(reserved-connection-pool
|
|
|
|
|
(make-resource-pool
|
|
|
|
|
(lambda ()
|
|
|
|
|
(open-postgresql-connection
|
|
|
|
|
"web-reserved"
|
|
|
|
|
postgresql-statement-timeout))
|
|
|
|
|
(floor (/ postgresql-connections 2))
|
2024-08-14 20:12:19 +01:00
|
|
|
#:name "web-reserved"
|
2024-07-16 21:50:36 +01:00
|
|
|
#:idle-seconds 600
|
|
|
|
|
#:destructor
|
|
|
|
|
(lambda (conn)
|
2024-08-14 21:13:55 +01:00
|
|
|
(close-postgresql-connection conn "web-reserved"))
|
2025-02-04 13:19:01 +00:00
|
|
|
#:default-checkout-timeout 6
|
|
|
|
|
#:scheduler priority-scheduler)))
|
2024-07-16 21:50:36 +01:00
|
|
|
|
|
|
|
|
(let ((resource-pool-checkout-failures-metric
|
|
|
|
|
(make-counter-metric registry
|
|
|
|
|
"resource_pool_checkout_timeouts_total"
|
|
|
|
|
#:labels '(pool_name))))
|
2024-12-15 19:08:28 +00:00
|
|
|
(resource-pool-default-timeout-handler
|
2024-07-16 21:50:36 +01:00
|
|
|
(lambda (pool proc timeout)
|
|
|
|
|
(let ((pool-name
|
|
|
|
|
(cond
|
|
|
|
|
((eq? pool (connection-pool)) "normal")
|
|
|
|
|
((eq? pool (reserved-connection-pool)) "reserved")
|
|
|
|
|
(else #f))))
|
|
|
|
|
(when pool-name
|
|
|
|
|
(metric-increment
|
|
|
|
|
resource-pool-checkout-failures-metric
|
|
|
|
|
#:label-values `((pool_name . ,pool-name))))))))
|
|
|
|
|
|
2025-05-14 15:30:07 +01:00
|
|
|
(spawn-fiber
|
|
|
|
|
(lambda ()
|
|
|
|
|
(while (not (check-startup-completed startup-completed))
|
|
|
|
|
(sleep 1))
|
|
|
|
|
|
|
|
|
|
(call-with-resource-from-pool (background-connection-pool)
|
|
|
|
|
(lambda (conn)
|
|
|
|
|
(let ((build-ids
|
|
|
|
|
(select-background-processing-build-ids conn)))
|
|
|
|
|
(unless (null? build-ids)
|
|
|
|
|
(simple-format #t "processing ~A builds from the background queue\n"
|
|
|
|
|
(length build-ids)))
|
|
|
|
|
(for-each
|
|
|
|
|
(lambda (build-id)
|
|
|
|
|
(let ((status (select-latest-build-status-by-build-id
|
|
|
|
|
conn
|
|
|
|
|
build-id)))
|
|
|
|
|
(cond
|
|
|
|
|
((string=? status "succeeded")
|
|
|
|
|
(handle-removing-blocking-build-entries-for-successful-builds
|
|
|
|
|
conn
|
|
|
|
|
(list build-id)))
|
|
|
|
|
((string=? status "scheduled")
|
|
|
|
|
(handle-blocked-builds-entries-for-scheduled-builds
|
|
|
|
|
conn
|
|
|
|
|
(list build-id)))
|
|
|
|
|
((member status '("failed"
|
|
|
|
|
"failed-dependency"
|
|
|
|
|
"canceled"))
|
|
|
|
|
(handle-populating-blocked-builds-for-build-failures
|
|
|
|
|
conn
|
|
|
|
|
(list build-id)))))
|
|
|
|
|
(delete-background-processing-entries-for-build-ids
|
|
|
|
|
conn
|
|
|
|
|
build-id))
|
|
|
|
|
build-ids)))))
|
|
|
|
|
#:parallel? #t)
|
|
|
|
|
|
2024-07-16 21:50:36 +01:00
|
|
|
(spawn-fiber
|
|
|
|
|
(lambda ()
|
2024-11-05 21:11:02 +00:00
|
|
|
(while (not (check-startup-completed startup-completed))
|
|
|
|
|
(sleep 1))
|
|
|
|
|
|
2024-09-08 10:20:21 +01:00
|
|
|
(with-resource-from-pool (background-connection-pool) conn
|
2024-07-16 21:50:36 +01:00
|
|
|
(backfill-guix-revision-package-derivation-distribution-counts
|
|
|
|
|
conn)))
|
2025-01-29 16:31:50 +00:00
|
|
|
#:parallel? #t)
|
2024-07-16 21:50:36 +01:00
|
|
|
|
2024-07-29 11:24:24 +01:00
|
|
|
(let ((render-metrics (make-render-metrics registry)))
|
2024-12-15 19:08:28 +00:00
|
|
|
(run-knots-web-server
|
|
|
|
|
(lambda (request)
|
2025-03-10 21:45:16 +00:00
|
|
|
(with-exception-handler
|
|
|
|
|
(lambda (exn)
|
|
|
|
|
(when (resource-pool-timeout-error? exn)
|
|
|
|
|
(spawn-fiber
|
|
|
|
|
(lambda ()
|
|
|
|
|
(let* ((pool (resource-pool-timeout-error-pool exn))
|
|
|
|
|
(stats (resource-pool-stats pool)))
|
|
|
|
|
(simple-format (current-error-port)
|
|
|
|
|
"resource pool timeout error: ~A, ~A\n"
|
|
|
|
|
pool
|
|
|
|
|
stats)))))
|
2024-07-29 11:24:24 +01:00
|
|
|
|
2025-03-10 21:45:16 +00:00
|
|
|
(let ((path-components
|
|
|
|
|
mime-types
|
2025-04-27 11:40:44 +01:00
|
|
|
(request->path-components-and-mime-type request))
|
|
|
|
|
(pool-exn?
|
|
|
|
|
(or (resource-pool-timeout-error? exn)
|
|
|
|
|
(resource-pool-too-many-waiters-error? exn))))
|
2025-03-10 21:45:16 +00:00
|
|
|
(case (most-appropriate-mime-type
|
|
|
|
|
mime-types
|
|
|
|
|
'(text/html application/json))
|
|
|
|
|
((application/json)
|
|
|
|
|
(apply
|
|
|
|
|
values
|
|
|
|
|
(render-json `((error . ,(if (%show-error-details)
|
|
|
|
|
(simple-format #f "~A" exn)
|
|
|
|
|
#f)))
|
2025-04-27 11:40:44 +01:00
|
|
|
#:code (if pool-exn?
|
|
|
|
|
503
|
|
|
|
|
500))))
|
2025-03-10 21:45:16 +00:00
|
|
|
(else
|
|
|
|
|
(apply
|
|
|
|
|
values
|
|
|
|
|
(render-html #:sxml (error-page
|
|
|
|
|
(if (%show-error-details)
|
|
|
|
|
exn
|
|
|
|
|
#f))
|
2025-04-27 11:40:44 +01:00
|
|
|
#:code (if pool-exn?
|
|
|
|
|
503
|
|
|
|
|
500)))))))
|
2025-03-10 21:45:16 +00:00
|
|
|
(lambda ()
|
|
|
|
|
(with-exception-handler
|
|
|
|
|
(lambda (exn)
|
|
|
|
|
(let* ((error-string
|
|
|
|
|
(call-with-output-string
|
|
|
|
|
(lambda (port)
|
|
|
|
|
(simple-format
|
|
|
|
|
port
|
|
|
|
|
"exception when processing: ~A ~A\n"
|
|
|
|
|
(request-method request)
|
|
|
|
|
(uri-path (request-uri request)))
|
|
|
|
|
(print-backtrace-and-exception/knots
|
|
|
|
|
exn
|
|
|
|
|
#:port port)))))
|
|
|
|
|
(display error-string
|
|
|
|
|
(current-error-port)))
|
2025-01-29 16:31:50 +00:00
|
|
|
|
2025-03-10 21:45:16 +00:00
|
|
|
(raise-exception exn))
|
|
|
|
|
(lambda ()
|
|
|
|
|
(metric-increment requests-metric)
|
2025-02-03 12:18:54 +01:00
|
|
|
|
2025-03-10 21:45:16 +00:00
|
|
|
(let ((body (read-request-body request)))
|
|
|
|
|
(handler request finished? body controller
|
|
|
|
|
secret-key-base
|
|
|
|
|
startup-completed
|
|
|
|
|
render-metrics)))))
|
|
|
|
|
#:unwind? #t))
|
2025-01-29 16:34:16 +00:00
|
|
|
#:connection-buffer-size (expt 2 16)
|
2024-07-29 11:24:24 +01:00
|
|
|
#:host host
|
|
|
|
|
#:port port)))
|
2023-07-09 18:06:57 +01:00
|
|
|
#:unwind? #t)))
|
2024-02-13 09:59:39 +00:00
|
|
|
|
|
|
|
|
;; Guile sometimes just seems to stop listening on the port, so try
|
|
|
|
|
;; and detect this and quit
|
|
|
|
|
(spawn-port-monitoring-fiber port finished?)
|
|
|
|
|
|
2023-07-09 20:53:10 +01:00
|
|
|
(wait finished?))
|
2024-07-26 21:42:21 +01:00
|
|
|
#:hz 0
|
2024-08-14 21:42:11 +01:00
|
|
|
#:parallelism 4))
|
2023-07-09 16:52:35 +01:00
|
|
|
finished?)))
|