Merge pull request #6939 from annando/ap-delivery-failure
Archive inboxes after 5 days of delivery failures
This commit is contained in:
commit
1f3b9ab451
5 changed files with 144 additions and 14 deletions
|
@ -34,7 +34,7 @@
|
||||||
use Friendica\Database\DBA;
|
use Friendica\Database\DBA;
|
||||||
|
|
||||||
if (!defined('DB_UPDATE_VERSION')) {
|
if (!defined('DB_UPDATE_VERSION')) {
|
||||||
define('DB_UPDATE_VERSION', 1307);
|
define('DB_UPDATE_VERSION', 1308);
|
||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
|
@ -529,6 +529,21 @@ return [
|
||||||
"hook_file_function" => ["UNIQUE", "hook", "file", "function"],
|
"hook_file_function" => ["UNIQUE", "hook", "file", "function"],
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
|
"inbox-status" => [
|
||||||
|
"comment" => "Status of ActivityPub inboxes",
|
||||||
|
"fields" => [
|
||||||
|
"url" => ["type" => "varbinary(255)", "not null" => "1", "primary" => "1", "comment" => "URL of the inbox"],
|
||||||
|
"created" => ["type" => "datetime", "not null" => "1", "default" => DBA::NULL_DATETIME, "comment" => "Creation date of this entry"],
|
||||||
|
"success" => ["type" => "datetime", "not null" => "1", "default" => DBA::NULL_DATETIME, "comment" => "Date of the last successful delivery"],
|
||||||
|
"failure" => ["type" => "datetime", "not null" => "1", "default" => DBA::NULL_DATETIME, "comment" => "Date of the last failed delivery"],
|
||||||
|
"previous" => ["type" => "datetime", "not null" => "1", "default" => DBA::NULL_DATETIME, "comment" => "Previous delivery date"],
|
||||||
|
"archive" => ["type" => "boolean", "not null" => "1", "default" => "0", "comment" => "Is the inbox archived?"],
|
||||||
|
"shared" => ["type" => "boolean", "not null" => "1", "default" => "0", "comment" => "Is it a shared inbox?"]
|
||||||
|
],
|
||||||
|
"indexes" => [
|
||||||
|
"PRIMARY" => ["url"]
|
||||||
|
]
|
||||||
|
],
|
||||||
"intro" => [
|
"intro" => [
|
||||||
"comment" => "",
|
"comment" => "",
|
||||||
"fields" => [
|
"fields" => [
|
||||||
|
|
33
database.sql
33
database.sql
|
@ -1,6 +1,6 @@
|
||||||
-- ------------------------------------------
|
-- ------------------------------------------
|
||||||
-- Friendica 2019.03-dev (The Tazmans Flax-lily)
|
-- Friendica 2019.06-dev (Dalmatian Bellflower)
|
||||||
-- DB_UPDATE_VERSION 1300
|
-- DB_UPDATE_VERSION 1308
|
||||||
-- ------------------------------------------
|
-- ------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
|
@ -470,6 +470,20 @@ CREATE TABLE IF NOT EXISTS `hook` (
|
||||||
UNIQUE INDEX `hook_file_function` (`hook`,`file`,`function`)
|
UNIQUE INDEX `hook_file_function` (`hook`,`file`,`function`)
|
||||||
) DEFAULT COLLATE utf8mb4_general_ci COMMENT='addon hook registry';
|
) DEFAULT COLLATE utf8mb4_general_ci COMMENT='addon hook registry';
|
||||||
|
|
||||||
|
--
|
||||||
|
-- TABLE inbox-status
|
||||||
|
--
|
||||||
|
CREATE TABLE IF NOT EXISTS `inbox-status` (
|
||||||
|
`url` varbinary(255) NOT NULL COMMENT 'URL of the inbox',
|
||||||
|
`created` datetime NOT NULL DEFAULT '0001-01-01 00:00:00' COMMENT 'Creation date of this entry',
|
||||||
|
`success` datetime NOT NULL DEFAULT '0001-01-01 00:00:00' COMMENT 'Date of the last successful delivery',
|
||||||
|
`failure` datetime NOT NULL DEFAULT '0001-01-01 00:00:00' COMMENT 'Date of the last failed delivery',
|
||||||
|
`previous` datetime NOT NULL DEFAULT '0001-01-01 00:00:00' COMMENT 'Previous delivery date',
|
||||||
|
`archive` boolean NOT NULL DEFAULT '0' COMMENT 'Is the inbox archived?',
|
||||||
|
`shared` boolean NOT NULL DEFAULT '0' COMMENT 'Is it a shared inbox?',
|
||||||
|
PRIMARY KEY(`url`)
|
||||||
|
) DEFAULT COLLATE utf8mb4_general_ci COMMENT='Status of ActivityPub inboxes';
|
||||||
|
|
||||||
--
|
--
|
||||||
-- TABLE intro
|
-- TABLE intro
|
||||||
--
|
--
|
||||||
|
@ -879,7 +893,7 @@ CREATE TABLE IF NOT EXISTS `photo` (
|
||||||
`deny_gid` mediumtext COMMENT 'Access Control - list of denied groups',
|
`deny_gid` mediumtext COMMENT 'Access Control - list of denied groups',
|
||||||
`backend-class` tinytext COMMENT 'Storage backend class',
|
`backend-class` tinytext COMMENT 'Storage backend class',
|
||||||
`backend-ref` text COMMENT 'Storage backend data reference',
|
`backend-ref` text COMMENT 'Storage backend data reference',
|
||||||
`updated` datetime NOT NULL DEFAULT '0001-01-01 00:00:00' COMMENT 'edited timestamp',
|
`updated` datetime NOT NULL DEFAULT '0001-01-01 00:00:00' COMMENT '',
|
||||||
PRIMARY KEY(`id`),
|
PRIMARY KEY(`id`),
|
||||||
INDEX `contactid` (`contact-id`),
|
INDEX `contactid` (`contact-id`),
|
||||||
INDEX `uid_contactid` (`uid`,`contact-id`),
|
INDEX `uid_contactid` (`uid`,`contact-id`),
|
||||||
|
@ -1099,7 +1113,7 @@ CREATE TABLE IF NOT EXISTS `term` (
|
||||||
`global` boolean NOT NULL DEFAULT '0' COMMENT '',
|
`global` boolean NOT NULL DEFAULT '0' COMMENT '',
|
||||||
`uid` mediumint unsigned NOT NULL DEFAULT 0 COMMENT 'User id',
|
`uid` mediumint unsigned NOT NULL DEFAULT 0 COMMENT 'User id',
|
||||||
PRIMARY KEY(`tid`),
|
PRIMARY KEY(`tid`),
|
||||||
INDEX `term_type` (`term`(64), `type`),
|
INDEX `term_type` (`term`(64),`type`),
|
||||||
INDEX `oid_otype_type_term` (`oid`,`otype`,`type`,`term`(32)),
|
INDEX `oid_otype_type_term` (`oid`,`otype`,`type`,`term`(32)),
|
||||||
INDEX `uid_otype_type_term_global_created` (`uid`,`otype`,`type`,`term`(32),`global`,`created`),
|
INDEX `uid_otype_type_term_global_created` (`uid`,`otype`,`type`,`term`(32),`global`,`created`),
|
||||||
INDEX `uid_otype_type_url` (`uid`,`otype`,`type`,`url`(64)),
|
INDEX `uid_otype_type_url` (`uid`,`otype`,`type`,`url`(64)),
|
||||||
|
@ -1271,13 +1285,12 @@ CREATE TABLE IF NOT EXISTS `workerqueue` (
|
||||||
`retrial` tinyint NOT NULL DEFAULT 0 COMMENT 'Retrial counter',
|
`retrial` tinyint NOT NULL DEFAULT 0 COMMENT 'Retrial counter',
|
||||||
`done` boolean NOT NULL DEFAULT '0' COMMENT 'Marked 1 when the task was done - will be deleted later',
|
`done` boolean NOT NULL DEFAULT '0' COMMENT 'Marked 1 when the task was done - will be deleted later',
|
||||||
PRIMARY KEY(`id`),
|
PRIMARY KEY(`id`),
|
||||||
INDEX `pid` (`pid`),
|
INDEX `done_parameter` (`done`,`parameter`(64)),
|
||||||
INDEX `parameter` (`parameter`(64)),
|
INDEX `done_executed` (`done`,`executed`),
|
||||||
INDEX `priority_created_next_try` (`priority`,`created`,`next_try`),
|
INDEX `done_priority_created` (`done`,`priority`,`created`),
|
||||||
INDEX `done_priority_executed_next_try` (`done`,`priority`,`executed`,`next_try`),
|
|
||||||
INDEX `done_executed_next_try` (`done`,`executed`,`next_try`),
|
|
||||||
INDEX `done_priority_next_try` (`done`,`priority`,`next_try`),
|
INDEX `done_priority_next_try` (`done`,`priority`,`next_try`),
|
||||||
INDEX `done_next_try` (`done`,`next_try`)
|
INDEX `done_pid_next_try` (`done`,`pid`,`next_try`),
|
||||||
|
INDEX `done_pid_priority_created` (`done`,`pid`,`priority`,`created`)
|
||||||
) DEFAULT COLLATE utf8mb4_general_ci COMMENT='Background tasks queue entries';
|
) DEFAULT COLLATE utf8mb4_general_ci COMMENT='Background tasks queue entries';
|
||||||
|
|
||||||
--
|
--
|
||||||
|
|
|
@ -123,11 +123,14 @@ class APContact extends BaseObject
|
||||||
$apcontact['following'] = JsonLD::fetchElement($compacted, 'as:following', '@id');
|
$apcontact['following'] = JsonLD::fetchElement($compacted, 'as:following', '@id');
|
||||||
$apcontact['followers'] = JsonLD::fetchElement($compacted, 'as:followers', '@id');
|
$apcontact['followers'] = JsonLD::fetchElement($compacted, 'as:followers', '@id');
|
||||||
$apcontact['inbox'] = JsonLD::fetchElement($compacted, 'ldp:inbox', '@id');
|
$apcontact['inbox'] = JsonLD::fetchElement($compacted, 'ldp:inbox', '@id');
|
||||||
|
self::unarchiveInbox($apcontact['inbox'], false);
|
||||||
|
|
||||||
$apcontact['outbox'] = JsonLD::fetchElement($compacted, 'as:outbox', '@id');
|
$apcontact['outbox'] = JsonLD::fetchElement($compacted, 'as:outbox', '@id');
|
||||||
|
|
||||||
$apcontact['sharedinbox'] = '';
|
$apcontact['sharedinbox'] = '';
|
||||||
if (!empty($compacted['as:endpoints'])) {
|
if (!empty($compacted['as:endpoints'])) {
|
||||||
$apcontact['sharedinbox'] = JsonLD::fetchElement($compacted['as:endpoints'], 'as:sharedInbox', '@id');
|
$apcontact['sharedinbox'] = JsonLD::fetchElement($compacted['as:endpoints'], 'as:sharedInbox', '@id');
|
||||||
|
self::unarchiveInbox($apcontact['sharedinbox'], true);
|
||||||
}
|
}
|
||||||
|
|
||||||
$apcontact['nick'] = JsonLD::fetchElement($compacted, 'as:preferredUsername');
|
$apcontact['nick'] = JsonLD::fetchElement($compacted, 'as:preferredUsername');
|
||||||
|
@ -231,4 +234,27 @@ class APContact extends BaseObject
|
||||||
|
|
||||||
return $apcontact;
|
return $apcontact;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unarchive inboxes
|
||||||
|
*
|
||||||
|
* @param string $url inbox url
|
||||||
|
*/
|
||||||
|
private static function unarchiveInbox($url, $shared)
|
||||||
|
{
|
||||||
|
if (empty($url)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$now = DateTimeFormat::utcNow();
|
||||||
|
|
||||||
|
$fields = ['archive' => false, 'success' => $now, 'shared' => $shared];
|
||||||
|
|
||||||
|
if (!DBA::exists('inbox-status', ['url' => $url])) {
|
||||||
|
$fields = array_merge($fields, ['url' => $url, 'created' => $now]);
|
||||||
|
DBA::insert('inbox-status', $fields);
|
||||||
|
} else {
|
||||||
|
DBA::update('inbox-status', $fields, ['url' => $url]);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -465,6 +465,18 @@ class Transmitter
|
||||||
return $receivers;
|
return $receivers;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if an inbox is archived
|
||||||
|
*
|
||||||
|
* @param string $url Inbox url
|
||||||
|
*
|
||||||
|
* @return boolean "true" if inbox is archived
|
||||||
|
*/
|
||||||
|
private static function archivedInbox($url)
|
||||||
|
{
|
||||||
|
return DBA::exists('inbox-status', ['url' => $url, 'archive' => true]);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fetches a list of inboxes of followers of a given user
|
* Fetches a list of inboxes of followers of a given user
|
||||||
*
|
*
|
||||||
|
@ -506,7 +518,9 @@ class Transmitter
|
||||||
} else {
|
} else {
|
||||||
$target = $profile['sharedinbox'];
|
$target = $profile['sharedinbox'];
|
||||||
}
|
}
|
||||||
$inboxes[$target] = $target;
|
if (!self::archivedInbox($target)) {
|
||||||
|
$inboxes[$target] = $target;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DBA::close($contacts);
|
DBA::close($contacts);
|
||||||
|
@ -563,7 +577,9 @@ class Transmitter
|
||||||
} else {
|
} else {
|
||||||
$target = $profile['sharedinbox'];
|
$target = $profile['sharedinbox'];
|
||||||
}
|
}
|
||||||
$inboxes[$target] = $target;
|
if (!self::archivedInbox($target)) {
|
||||||
|
$inboxes[$target] = $target;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
*/
|
*/
|
||||||
namespace Friendica\Util;
|
namespace Friendica\Util;
|
||||||
|
|
||||||
|
use Friendica\Database\DBA;
|
||||||
use Friendica\Core\Config;
|
use Friendica\Core\Config;
|
||||||
use Friendica\Core\Logger;
|
use Friendica\Core\Logger;
|
||||||
use Friendica\Model\User;
|
use Friendica\Model\User;
|
||||||
|
@ -314,7 +315,66 @@ class HTTPSignature
|
||||||
|
|
||||||
Logger::log('Transmit to ' . $target . ' returned ' . $return_code, Logger::DEBUG);
|
Logger::log('Transmit to ' . $target . ' returned ' . $return_code, Logger::DEBUG);
|
||||||
|
|
||||||
return ($return_code >= 200) && ($return_code <= 299);
|
$success = ($return_code >= 200) && ($return_code <= 299);
|
||||||
|
|
||||||
|
self::setInboxStatus($target, $success);
|
||||||
|
|
||||||
|
return $success;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Set the delivery status for a given inbox
|
||||||
|
*
|
||||||
|
* @param string $url The URL of the inbox
|
||||||
|
* @param boolean $success Transmission status
|
||||||
|
*/
|
||||||
|
static private function setInboxStatus($url, $success)
|
||||||
|
{
|
||||||
|
$now = DateTimeFormat::utcNow();
|
||||||
|
|
||||||
|
$status = DBA::selectFirst('inbox-status', [], ['url' => $url]);
|
||||||
|
if (!DBA::isResult($status)) {
|
||||||
|
DBA::insert('inbox-status', ['url' => $url, 'created' => $now]);
|
||||||
|
$status = DBA::selectFirst('inbox-status', [], ['url' => $url]);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($success) {
|
||||||
|
$fields = ['success' => $now];
|
||||||
|
} else {
|
||||||
|
$fields = ['failure' => $now];
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($status['failure'] > DBA::NULL_DATETIME) {
|
||||||
|
$new_previous_stamp = strtotime($status['failure']);
|
||||||
|
$old_previous_stamp = strtotime($status['previous']);
|
||||||
|
|
||||||
|
// Only set "previous" with at least one day difference.
|
||||||
|
// We use this to assure to not accidentally archive too soon.
|
||||||
|
if (($new_previous_stamp - $old_previous_stamp) >= 86400) {
|
||||||
|
$fields['previous'] = $status['failure'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$success) {
|
||||||
|
if ($status['success'] <= DBA::NULL_DATETIME) {
|
||||||
|
$stamp1 = strtotime($status['created']);
|
||||||
|
} else {
|
||||||
|
$stamp1 = strtotime($status['success']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$stamp2 = strtotime($now);
|
||||||
|
$previous_stamp = strtotime($status['previous']);
|
||||||
|
|
||||||
|
// Archive the inbox when there had been failures for five days.
|
||||||
|
// Additionally ensure that at least one previous attempt has to be in between.
|
||||||
|
if ((($stamp2 - $stamp1) >= 86400 * 5) && ($previous_stamp > $stamp1)) {
|
||||||
|
$fields['archive'] = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$fields['archive'] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
DBA::update('inbox-status', $fields, ['url' => $url]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
Loading…
Reference in a new issue