From 79b0d0b9ccdb7e6e61e73adc420fe97fafb32178 Mon Sep 17 00:00:00 2001 From: jvoisin Date: Fri, 4 Jul 2025 17:18:28 +0200 Subject: [PATCH] feat(integration): add integration with archive.org MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Tested locally: ```console $ Tue 26 Aug 17:34:05 CEST 2025 $ go build && ./miniflux.app -c ./config.ini -debug level=DEBUG msg="Starting daemon..." level=DEBUG msg="Starting background scheduler..." level=DEBUG msg="Worker started" worker_id=15 level=DEBUG msg="Worker started" worker_id=0 […] level=DEBUG msg="Incoming request" client_ip=127.0.0.1 request.method=POST request.uri=/entry/save/29773 request.protocol=HTTP/1.1 request.execution_time=5.57385ms level=DEBUG msg="Sending entry to archive.org" user_id=1 entry_id=29773 entry_url=https://sumnerevans.com/portfolio/ level=DEBUG msg="Sending entry to archive.org" title=Portfolio url=https://sumnerevans.com/portfolio/ ^C $ curl -I -H "User-Agent: Mozilla" https://web.archive.org/web/20250826153413/https://sumnerevans.com/portfolio/ | grep orig-date x-archive-orig-date: Tue, 26 Aug 2025 15:34:13 GMT $ ``` --- internal/database/migrations.go | 7 +++ internal/integration/archiveorg/archiveorg.go | 43 +++++++++++++++++++ internal/integration/integration.go | 12 +++++- internal/locale/translations/de_DE.json | 1 + internal/locale/translations/el_EL.json | 1 + internal/locale/translations/en_US.json | 1 + internal/locale/translations/es_ES.json | 1 + internal/locale/translations/fi_FI.json | 1 + internal/locale/translations/fr_FR.json | 1 + internal/locale/translations/hi_IN.json | 1 + internal/locale/translations/id_ID.json | 1 + internal/locale/translations/it_IT.json | 1 + internal/locale/translations/ja_JP.json | 1 + .../locale/translations/nan_Latn_pehoeji.json | 1 + internal/locale/translations/nl_NL.json | 1 + internal/locale/translations/pl_PL.json | 1 + internal/locale/translations/pt_BR.json | 1 + internal/locale/translations/ro_RO.json | 1 + internal/locale/translations/ru_RU.json | 1 + internal/locale/translations/tr_TR.json | 1 + internal/locale/translations/uk_UA.json | 1 + internal/locale/translations/zh_CN.json | 1 + internal/locale/translations/zh_TW.json | 1 + internal/model/integration.go | 1 + internal/storage/integration.go | 13 ++++-- .../templates/views/integrations.html | 12 ++++++ internal/ui/form/integration.go | 3 ++ internal/ui/integration_show.go | 1 + 28 files changed, 107 insertions(+), 5 deletions(-) create mode 100644 internal/integration/archiveorg/archiveorg.go diff --git a/internal/database/migrations.go b/internal/database/migrations.go index 2f6eaa35..f6f92956 100644 --- a/internal/database/migrations.go +++ b/internal/database/migrations.go @@ -1341,4 +1341,11 @@ var migrations = [...]func(tx *sql.Tx) error{ return nil }, + func(tx *sql.Tx) (err error) { + sql := ` + ALTER TABLE integrations ADD COLUMN archiveorg_enabled bool default 'f' + ` + _, err = tx.Exec(sql) + return err + }, } diff --git a/internal/integration/archiveorg/archiveorg.go b/internal/integration/archiveorg/archiveorg.go new file mode 100644 index 00000000..f86a37ec --- /dev/null +++ b/internal/integration/archiveorg/archiveorg.go @@ -0,0 +1,43 @@ +// SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved. +// SPDX-License-Identifier: Apache-2.0 + +package archiveorg + +import ( + "log/slog" + "net/http" + "net/url" +) + +// See https://docs.google.com/document/d/1Nsv52MvSjbLb2PCpHlat0gkzw0EvtSgpKHu4mk0MnrA/edit?tab=t.0 +const options = "delay_wb_availability=1&if_not_archived_within=15d" + +type Client struct{} + +func NewClient() *Client { + return &Client{} +} + +func (c *Client) SendURL(entryURL, title string) { + // We're using a goroutine here as submissions to archive.org might take a long time + // and trigger a timeout on miniflux' side. + go func(entryURL string) { + res, err := http.Get("https://web.archive.org/save/" + url.QueryEscape(entryURL) + "?" + options) + if err != nil { + slog.Error("archiveorg: unable to send request: %v", + slog.Any("err", err), + slog.String("title", title), + slog.String("url", entryURL), + ) + return + } + if res.StatusCode > 299 { + slog.Error("archiveorg: failed with status code", + slog.String("title", title), + slog.String("url", entryURL), + slog.Int("code", res.StatusCode), + ) + } + res.Body.Close() + }(entryURL) +} diff --git a/internal/integration/integration.go b/internal/integration/integration.go index bec8b436..1f24ba3d 100644 --- a/internal/integration/integration.go +++ b/internal/integration/integration.go @@ -7,6 +7,7 @@ import ( "log/slog" "miniflux.app/v2/internal/integration/apprise" + "miniflux.app/v2/internal/integration/archiveorg" "miniflux.app/v2/internal/integration/betula" "miniflux.app/v2/internal/integration/cubox" "miniflux.app/v2/internal/integration/discord" @@ -398,6 +399,16 @@ func SendEntry(entry *model.Entry, userIntegrations *model.Integration) { } } + if userIntegrations.ArchiveorgEnabled { + slog.Debug("Sending entry to archive.org", + slog.Int64("user_id", userIntegrations.UserID), + slog.Int64("entry_id", entry.ID), + slog.String("entry_url", entry.URL), + ) + + archiveorg.NewClient().SendURL(entry.URL, entry.Title) + } + if userIntegrations.WebhookEnabled { var webhookURL string if entry.Feed != nil && entry.Feed.WebhookURL != "" { @@ -506,7 +517,6 @@ func PushEntries(feed *model.Feed, entries model.Entries, userIntegrations *mode ) } } - if userIntegrations.WebhookEnabled { var webhookURL string if feed.WebhookURL != "" { diff --git a/internal/locale/translations/de_DE.json b/internal/locale/translations/de_DE.json index 2d206654..3ea1a76d 100644 --- a/internal/locale/translations/de_DE.json +++ b/internal/locale/translations/de_DE.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Webhook-URL überschreiben", "form.import.label.file": "OPML-Datei", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Artikel zu archive.org pushen", "form.integration.apprise_activate": "Artikel zu Apprise pushen", "form.integration.apprise_services_url": "Kommaseparierte Liste von Apprise-Dienst-URLs", "form.integration.apprise_url": "Apprise-API-URL", diff --git a/internal/locale/translations/el_EL.json b/internal/locale/translations/el_EL.json index 9b14f075..03d0cc9f 100644 --- a/internal/locale/translations/el_EL.json +++ b/internal/locale/translations/el_EL.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Παράκαμψη διεύθυνσης URL webhook", "form.import.label.file": "Αρχείο OPML", "form.import.label.url": "Διεύθυνση URL", + "form.integration.archiveorg_activate": "Προώθηση καταχωρήσεων στο archive.org", "form.integration.apprise_activate": "Προώθηση καταχωρήσεων στο Apprise", "form.integration.apprise_services_url": "Λίστα διευθύνσεων URL υπηρεσιών Apprise διαχωρισμένων με κόμμα", "form.integration.apprise_url": "Διεύθυνση URL API Apprise", diff --git a/internal/locale/translations/en_US.json b/internal/locale/translations/en_US.json index 35bff0f4..c6b87dcb 100644 --- a/internal/locale/translations/en_US.json +++ b/internal/locale/translations/en_US.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Override webhook url", "form.import.label.file": "OPML file", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Push entries to archive.org", "form.integration.apprise_activate": "Push entries to Apprise", "form.integration.apprise_services_url": "Comma separated list of Apprise service URLs", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/es_ES.json b/internal/locale/translations/es_ES.json index 80de07fb..b15c5c71 100644 --- a/internal/locale/translations/es_ES.json +++ b/internal/locale/translations/es_ES.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Invalidar la URL del webhook", "form.import.label.file": "Archivo OPML", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Enviar entradas a archive.org", "form.integration.apprise_activate": "Enviar artículos a Apprise", "form.integration.apprise_services_url": "Lista separada por comas de las URL del servicio Apprise", "form.integration.apprise_url": "URL de la API de Apprise", diff --git a/internal/locale/translations/fi_FI.json b/internal/locale/translations/fi_FI.json index fbe3a909..7fe44fce 100644 --- a/internal/locale/translations/fi_FI.json +++ b/internal/locale/translations/fi_FI.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Override webhook url", "form.import.label.file": "OPML-tiedosto", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Työnnä merkinnät osoitteeseen archive.org", "form.integration.apprise_activate": "Push entries to Apprise", "form.integration.apprise_services_url": "Comma separated list of Apprise service URLs", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/fr_FR.json b/internal/locale/translations/fr_FR.json index 97a95cd9..9f6bccbf 100644 --- a/internal/locale/translations/fr_FR.json +++ b/internal/locale/translations/fr_FR.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Remplacer l'URL du webhook", "form.import.label.file": "Fichier OPML", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Envoyer les articles vers archive.org", "form.integration.apprise_activate": "Envoyer les articles vers Apprise", "form.integration.apprise_services_url": "Liste des services Apprise séparés par des virgules", "form.integration.apprise_url": "URL de l'API Apprise", diff --git a/internal/locale/translations/hi_IN.json b/internal/locale/translations/hi_IN.json index 0a50fd66..d331b70f 100644 --- a/internal/locale/translations/hi_IN.json +++ b/internal/locale/translations/hi_IN.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Override webhook url", "form.import.label.file": "ओपीएमएल फ़ाइल", "form.import.label.url": "यूआरएल", + "form.integration.archiveorg_activate": "प्रविष्टियों को archive.org पर भेजें", "form.integration.apprise_activate": "Push entries to Apprise", "form.integration.apprise_services_url": "Comma separated list of Apprise service URLs", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/id_ID.json b/internal/locale/translations/id_ID.json index cb26e018..8e18c4b1 100644 --- a/internal/locale/translations/id_ID.json +++ b/internal/locale/translations/id_ID.json @@ -209,6 +209,7 @@ "form.feed.label.webhook_url": "Timpa URL Webhook", "form.import.label.file": "Berkas OPML", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Push entries to archive.org", "form.integration.apprise_activate": "Kirim artikel ke Apprise", "form.integration.apprise_services_url": "Daftar yang dipisahkan koma untuk URL layanan Apprise", "form.integration.apprise_url": "URL API Apprise", diff --git a/internal/locale/translations/it_IT.json b/internal/locale/translations/it_IT.json index f5dae9f4..005bb400 100644 --- a/internal/locale/translations/it_IT.json +++ b/internal/locale/translations/it_IT.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Override webhook url", "form.import.label.file": "File OPML", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Invia le voci ad archive.org", "form.integration.apprise_activate": "Push entries to Apprise", "form.integration.apprise_services_url": "Comma separated list of Apprise service URLs", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/ja_JP.json b/internal/locale/translations/ja_JP.json index 7e0829ea..3f6f0349 100644 --- a/internal/locale/translations/ja_JP.json +++ b/internal/locale/translations/ja_JP.json @@ -209,6 +209,7 @@ "form.feed.label.webhook_url": "Override webhook url", "form.import.label.file": "OPML ファイル", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "エントリーをarchive.orgにプッシュする", "form.integration.apprise_activate": "Push entries to Apprise", "form.integration.apprise_services_url": "Comma separated list of Apprise service URLs", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/nan_Latn_pehoeji.json b/internal/locale/translations/nan_Latn_pehoeji.json index 63627ffe..79384c2a 100644 --- a/internal/locale/translations/nan_Latn_pehoeji.json +++ b/internal/locale/translations/nan_Latn_pehoeji.json @@ -209,6 +209,7 @@ "form.feed.label.webhook_url": "Ngī kái webhook bāng-chí", "form.import.label.file": "OPML tóng-àn", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Push entries to archive.org", "form.integration.apprise_activate": "Thui sàng siau-sit khì Apprise", "form.integration.apprise_services_url": "Iōng tō͘-tiám keh khui ê Apprise ho̍k-bū bāng-chí lia̍t-pió", "form.integration.apprise_url": "Apprise API bāng-chí", diff --git a/internal/locale/translations/nl_NL.json b/internal/locale/translations/nl_NL.json index 1562837a..9cbddcd4 100644 --- a/internal/locale/translations/nl_NL.json +++ b/internal/locale/translations/nl_NL.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Overschrijf webhook URL", "form.import.label.file": "OPML-bestand", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Push entries to archive.org", "form.integration.apprise_activate": "Artikelen opslaan in Apprise", "form.integration.apprise_services_url": "Door komma's gescheiden lijst van Apprise service URL's", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/pl_PL.json b/internal/locale/translations/pl_PL.json index 5a74e039..4025ceb4 100644 --- a/internal/locale/translations/pl_PL.json +++ b/internal/locale/translations/pl_PL.json @@ -215,6 +215,7 @@ "form.feed.label.webhook_url": "Zastąp adres URL webhooka", "form.import.label.file": "Plik OPML", "form.import.label.url": "Adres URL", + "form.integration.archiveorg_activate": "Prześlij wpisy do archive.org", "form.integration.apprise_activate": "Przesyłaj wpisy do Apprise", "form.integration.apprise_services_url": "Oddzielona przecinkami lista adresów URL usługi Apprise", "form.integration.apprise_url": "Adres URL API Apprise", diff --git a/internal/locale/translations/pt_BR.json b/internal/locale/translations/pt_BR.json index 0c97a68e..dddfda8c 100644 --- a/internal/locale/translations/pt_BR.json +++ b/internal/locale/translations/pt_BR.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Sobrescrever URL do webhook", "form.import.label.file": "Arquivo OPML", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Enviar itens para o archive.org", "form.integration.apprise_activate": "Enviar itens para o Apprise", "form.integration.apprise_services_url": "Lista de URLs de serviços Apprise separadas por vírgula", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/ro_RO.json b/internal/locale/translations/ro_RO.json index 2d1f6714..8a36225a 100644 --- a/internal/locale/translations/ro_RO.json +++ b/internal/locale/translations/ro_RO.json @@ -215,6 +215,7 @@ "form.feed.label.webhook_url": "URL Webhook (pentru a primi notificări despre evenimentele de intrare)", "form.import.label.file": "Fișier OPML", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Trimite înregistrările pe archive.org", "form.integration.apprise_activate": "Trimite înregistrările pe Apprise", "form.integration.apprise_services_url": "URL-uri separate de virgulă cu servicii Apprise", "form.integration.apprise_url": "URL API Apprise", diff --git a/internal/locale/translations/ru_RU.json b/internal/locale/translations/ru_RU.json index e3bd71d5..bbe9ea22 100644 --- a/internal/locale/translations/ru_RU.json +++ b/internal/locale/translations/ru_RU.json @@ -215,6 +215,7 @@ "form.feed.label.webhook_url": "Переопределить URL вебхука", "form.import.label.file": "OPML файл", "form.import.label.url": "Ссылка", + "form.integration.archiveorg_activate": "TОтправить статьи в archive.org", "form.integration.apprise_activate": "Отправить статьи в Apprise", "form.integration.apprise_services_url": "Список ссылок сервисов Apprise, разделенный запятой", "form.integration.apprise_url": "Ссылка на Apprise API", diff --git a/internal/locale/translations/tr_TR.json b/internal/locale/translations/tr_TR.json index 2e85d248..e9ff7743 100644 --- a/internal/locale/translations/tr_TR.json +++ b/internal/locale/translations/tr_TR.json @@ -212,6 +212,7 @@ "form.feed.label.webhook_url": "Webhook URL'sini geçersiz kıl", "form.import.label.file": "OPML dosyası", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "Makaleleri archive.org'a gönder", "form.integration.apprise_activate": "Makaleleri Apprise'a gönder", "form.integration.apprise_services_url": "Apprise hizmet URL'lerinin virgülle ayrılmış listesi", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/uk_UA.json b/internal/locale/translations/uk_UA.json index 14d6d6ab..85a25aa0 100644 --- a/internal/locale/translations/uk_UA.json +++ b/internal/locale/translations/uk_UA.json @@ -215,6 +215,7 @@ "form.feed.label.webhook_url": "Перевизначити URL вебхука", "form.import.label.file": "Файл OPML", "form.import.label.url": "URL-адреса", + "form.integration.archiveorg_activate": "Надсилати записи у archive.org", "form.integration.apprise_activate": "Надсилати записи у Apprise", "form.integration.apprise_services_url": "Список URL сервісів Apprise, розділених комами", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/zh_CN.json b/internal/locale/translations/zh_CN.json index bc355fa0..c467ce03 100644 --- a/internal/locale/translations/zh_CN.json +++ b/internal/locale/translations/zh_CN.json @@ -209,6 +209,7 @@ "form.feed.label.webhook_url": "覆盖 Webhook URL", "form.import.label.file": "OPML 文件", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "将新条目推送到 archive.org", "form.integration.apprise_activate": "将新条目推送到 Apprise", "form.integration.apprise_services_url": "使用逗号分隔的 Apprise 服务 URL 列表", "form.integration.apprise_url": "Apprise API URL", diff --git a/internal/locale/translations/zh_TW.json b/internal/locale/translations/zh_TW.json index bad9c8a4..2678cc0f 100644 --- a/internal/locale/translations/zh_TW.json +++ b/internal/locale/translations/zh_TW.json @@ -209,6 +209,7 @@ "form.feed.label.webhook_url": "覆蓋webhook URL", "form.import.label.file": "OPML 檔案", "form.import.label.url": "URL", + "form.integration.archiveorg_activate": "推送文章到 archive.org", "form.integration.apprise_activate": "推送文章到 Apprise", "form.integration.apprise_services_url": "使用逗號分隔的 Apprise 服務網址列表", "form.integration.apprise_url": "Apprise API 網址", diff --git a/internal/model/integration.go b/internal/model/integration.go index bff86357..e18c5794 100644 --- a/internal/model/integration.go +++ b/internal/model/integration.go @@ -123,4 +123,5 @@ type Integration struct { PushoverToken string PushoverDevice string PushoverPrefix string + ArchiveorgEnabled bool } diff --git a/internal/storage/integration.go b/internal/storage/integration.go index 761a8dd8..da89fc4b 100644 --- a/internal/storage/integration.go +++ b/internal/storage/integration.go @@ -226,7 +226,8 @@ func (s *Storage) Integration(userID int64) (*model.Integration, error) { linktaco_api_token, linktaco_org_slug, linktaco_tags, - linktaco_visibility + linktaco_visibility, + archiveorg_enabled FROM integrations WHERE @@ -352,6 +353,7 @@ func (s *Storage) Integration(userID int64) (*model.Integration, error) { &integration.LinktacoOrgSlug, &integration.LinktacoTags, &integration.LinktacoVisibility, + &integration.ArchiveorgEnabled, ) switch { case err == sql.ErrNoRows: @@ -485,9 +487,10 @@ func (s *Storage) UpdateIntegration(integration *model.Integration) error { linktaco_api_token=$114, linktaco_org_slug=$115, linktaco_tags=$116, - linktaco_visibility=$117 + linktaco_visibility=$117, + archiveorg_enabled=$118 WHERE - user_id=$118 + user_id=$119 ` _, err := s.db.Exec( query, @@ -608,6 +611,7 @@ func (s *Storage) UpdateIntegration(integration *model.Integration) error { integration.LinktacoOrgSlug, integration.LinktacoTags, integration.LinktacoVisibility, + integration.ArchiveorgEnabled, integration.UserID, ) @@ -651,7 +655,8 @@ func (s *Storage) HasSaveEntry(userID int64) (result bool) { betula_enabled='t' OR cubox_enabled='t' OR discord_enabled='t' OR - slack_enabled='t' + slack_enabled='t' OR + archiveorg_enabled='t' ) ` if err := s.db.QueryRow(query, userID).Scan(&result); err != nil { diff --git a/internal/template/templates/views/integrations.html b/internal/template/templates/views/integrations.html index 8ce2b933..0bb72204 100644 --- a/internal/template/templates/views/integrations.html +++ b/internal/template/templates/views/integrations.html @@ -15,6 +15,18 @@ {{ end }} +
+ Archive.org +
+ +
+ +
+
+
+
Apprise
diff --git a/internal/ui/form/integration.go b/internal/ui/form/integration.go index a5b03c18..04ba208d 100644 --- a/internal/ui/form/integration.go +++ b/internal/ui/form/integration.go @@ -129,6 +129,7 @@ type IntegrationForm struct { PushoverToken string PushoverDevice string PushoverPrefix string + ArchiveorgEnabled bool } // Merge copy form values to the model. @@ -247,6 +248,7 @@ func (i IntegrationForm) Merge(integration *model.Integration) { integration.PushoverToken = i.PushoverToken integration.PushoverDevice = i.PushoverDevice integration.PushoverPrefix = i.PushoverPrefix + integration.ArchiveorgEnabled = i.ArchiveorgEnabled } // NewIntegrationForm returns a new IntegrationForm. @@ -368,6 +370,7 @@ func NewIntegrationForm(r *http.Request) *IntegrationForm { PushoverToken: r.FormValue("pushover_token"), PushoverDevice: r.FormValue("pushover_device"), PushoverPrefix: r.FormValue("pushover_prefix"), + ArchiveorgEnabled: r.FormValue("archiveorg_enabled") == "1", } } diff --git a/internal/ui/integration_show.go b/internal/ui/integration_show.go index 13cc18d3..6a7af48f 100644 --- a/internal/ui/integration_show.go +++ b/internal/ui/integration_show.go @@ -142,6 +142,7 @@ func (h *handler) showIntegrationPage(w http.ResponseWriter, r *http.Request) { PushoverToken: integration.PushoverToken, PushoverDevice: integration.PushoverDevice, PushoverPrefix: integration.PushoverPrefix, + ArchiveorgEnabled: integration.ArchiveorgEnabled, } sess := session.New(h.store, request.SessionID(r))