aboutsummaryrefslogtreecommitdiff
;;; guile-email --- Guile email parser
;;; Copyright © 2021 Arun Isaac <arunisaac@systemreboot.net>
;;;
;;; This file is part of guile-email.
;;;
;;; guile-email 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.
;;;
;;; guile-email 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 guile-email.  If not, see
;;; <http://www.gnu.org/licenses/>.

;;; Commentary:

;; This script pulls archives of specified GNU mailing lists
;; (currently guix-commits and guix-patches). If a partial archive
;; already exists in the current directory, only the newer messages
;; are downloaded.

;;; Code:

(use-modules (ice-9 format)
             (rnrs io ports)
             (srfi srfi-1)
             (srfi srfi-9 gnu)
             (srfi srfi-19)
             (srfi srfi-41)
             (srfi srfi-71)
             (web client)
             (web response))

(define-immutable-record-type <archive>
  (make-archive mailing-list year month size)
  archive?
  (mailing-list archive-mailing-list)
  (year archive-year set-archive-year)
  (month archive-month set-archive-month)
  (size archive-size set-archive-size))

(define (archive->uri archive)
  (format #f "https://lists.gnu.org/archive/mbox/~a/~a-~2,'0d"
          (archive-mailing-list archive)
          (archive-year archive)
          (archive-month archive)))

(define (initialize-archive-size archive)
  (let ((response body (http-head (archive->uri archive))))
    (set-archive-size archive (assoc-ref (response-headers response) 'content-length))))

(define (previous-archive archive)
  (initialize-archive-size
   (case (archive-month archive)
     ((1) (set-archive-year
           (set-archive-month archive 12)
           (1- (archive-year archive))))
     (else (set-archive-month archive (1- (archive-month archive)))))))

(define (archive->filename archive)
  (format #f "~a/~a-~2,'0d.mbox"
          (archive-mailing-list archive)
          (archive-year archive)
          (archive-month archive)))

(define file-size
  (compose stat:size stat))

(define (out-of-sync? archive)
  (let ((filename (archive->filename archive)))
    (if (file-exists? filename)
        (< (file-size filename)
           (archive-size archive))
        #t)))

(define (pull-archive archive)
  (let ((file (archive->filename archive)))
    (format #t "Pulling archive ~a~%" (archive->filename archive))
    (call-with-port (open-file file "a")
      (lambda (port)
        (let ((response body (http-get (archive->uri archive)
                                       #:decode-body? #f
                                       #:headers `((range . (bytes (,(file-size file) . #f)))))))
          (put-bytevector port body))))))

(define (pull-mailing-list mailing-list)
  "Pull archives of MAILING-LIST to a directory of the same name."
  (define archives
    (let ((today (current-date)))
      (stream-unfold
       identity archive-size previous-archive
       (initialize-archive-size
        (make-archive mailing-list (date-year today) (date-month today) #f)))))

  (unless (file-exists? mailing-list)
    (mkdir mailing-list))
  (stream-for-each pull-archive (stream-take-while out-of-sync? archives)))

(pull-mailing-list "guix-commits")
(pull-mailing-list "guix-patches")
(pull-mailing-list "bug-guix")