forked from friendica/friendica-addons
Compare commits
94 commits
develop
...
mat/2023.0
Author | SHA1 | Date | |
---|---|---|---|
3da587f07e | |||
d05c5c6dc5 | |||
9ae80d7445 | |||
48f4dc0175 | |||
Michael | 3a2065b505 | ||
1fc7334639 | |||
025b362290 | |||
f780c6e5b2 | |||
893ab076e8 | |||
d18a905022 | |||
6a3175537e | |||
ce6a14e055 | |||
b7f4346211 | |||
52c287269f | |||
a0fbb7b9be | |||
adf160716c | |||
55dd0e84d8 | |||
8ad2b51492 | |||
061545c456 | |||
91e5b6337b | |||
1e6de4af53 | |||
16cf075e82 | |||
2abe0398ad | |||
fa71129205 | |||
0cbcbadec0 | |||
dee4f73b0a | |||
9cd1c2b72a | |||
dc9ebc732a | |||
25e033eb88 | |||
aa9fbc82bd | |||
2f5f240d78 | |||
2b56f3f25f | |||
003d7b6aa0 | |||
d997efe533 | |||
cf7da5a247 | |||
56c8ad70ae | |||
86ca889189 | |||
775047d3d9 | |||
92aecb3a1e | |||
45e42cbd9f | |||
8477a0f012 | |||
37797b5b34 | |||
6199e3af0e | |||
b9db44ba55 | |||
68f2dd3886 | |||
668590cffa | |||
e962de9425 | |||
1d8df0b95b | |||
912e24030d | |||
b93a203740 | |||
51ed5a3d5c | |||
542185285b | |||
12388eced8 | |||
e7fb24e986 | |||
b9f048c2a8 | |||
e090a286b1 | |||
91689cd798 | |||
3cdffe1fde | |||
6e5e06e303 | |||
67fc2a8491 | |||
9bf8602d8c | |||
592b28c09b | |||
7e1a495e5f | |||
d36fade822 | |||
351482464b | |||
c8cbd41161 | |||
e8ab4f3adb | |||
b1b9fd6af8 | |||
2d8e13d53d | |||
dbfc24d51f | |||
42314b6670 | |||
2ba05cc80c | |||
be68a4aa3c | |||
dbd00503aa | |||
3906813dcf | |||
043c515707 | |||
0c9db8383a | |||
04e57e4334 | |||
0963f0da4a | |||
4dc51d8f05 | |||
c5fb494552 | |||
615992810a | |||
ef6709d861 | |||
61e925630d | |||
10f7be958b | |||
034ed5fcd6 | |||
df7ea6c375 | |||
eb61f8f09a | |||
8b6a9c017a | |||
c9f4ad7405 | |||
738d1ab588 | |||
ae3fa6cea2 | |||
f453c15259 | |||
b994de3308 |
|
@ -180,5 +180,5 @@ function ifttt_message($uid, $item)
|
||||||
$link = hash('ripemd128', $item['msg']);
|
$link = hash('ripemd128', $item['msg']);
|
||||||
}
|
}
|
||||||
|
|
||||||
Post\Delayed::add($link, $post, Worker::PRIORITY_MEDIUM, Post\Delayed::PREPARED);
|
Post\Delayed::add($link, $post, Worker::PRIORITY_MEDIUM, Post\Delayed::UNPREPARED);
|
||||||
}
|
}
|
||||||
|
|
|
@ -416,7 +416,7 @@ function mailstream_send(string $message_id, array $item, array $user): bool
|
||||||
'$upstream' => DI::l10n()->t('Upstream'),
|
'$upstream' => DI::l10n()->t('Upstream'),
|
||||||
'$local' => DI::l10n()->t('Local'),
|
'$local' => DI::l10n()->t('Local'),
|
||||||
'$item' => $item]);
|
'$item' => $item]);
|
||||||
mailstream_html_wrap($mail->Body);
|
$mail->Body = mailstream_html_wrap($mail->Body);
|
||||||
if (!$mail->Send()) {
|
if (!$mail->Send()) {
|
||||||
throw new Exception($mail->ErrorInfo);
|
throw new Exception($mail->ErrorInfo);
|
||||||
}
|
}
|
||||||
|
@ -437,7 +437,8 @@ function mailstream_send(string $message_id, array $item, array $user): bool
|
||||||
* bbcode's output suitable for transmission, we try to break things
|
* bbcode's output suitable for transmission, we try to break things
|
||||||
* up so that lines are about 200 characters.
|
* up so that lines are about 200 characters.
|
||||||
*
|
*
|
||||||
* @param string $text text to word wrap - modified in-place
|
* @param string $text text to word wrap
|
||||||
|
* @return string wrapped text
|
||||||
*/
|
*/
|
||||||
function mailstream_html_wrap(string &$text)
|
function mailstream_html_wrap(string &$text)
|
||||||
{
|
{
|
||||||
|
@ -446,6 +447,7 @@ function mailstream_html_wrap(string &$text)
|
||||||
$lines[$i] = preg_replace('/ /', "\n", $lines[$i], 1);
|
$lines[$i] = preg_replace('/ /', "\n", $lines[$i], 1);
|
||||||
}
|
}
|
||||||
$text = implode($lines);
|
$text = implode($lines);
|
||||||
|
return $text;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
23
phototrack/database.sql
Normal file
23
phototrack/database.sql
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
CREATE TABLE IF NOT EXISTS `phototrack_photo_use` (
|
||||||
|
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
|
||||||
|
`resource-id` char(64) NOT NULL,
|
||||||
|
`table` char(64) NOT NULL,
|
||||||
|
`field` char(64) NOT NULL,
|
||||||
|
`row-id` int(11) NOT NULL,
|
||||||
|
`checked` timestamp NOT NULL DEFAULT now(),
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
INDEX `resource-id` (`resource-id`),
|
||||||
|
INDEX `row` (`table`,`field`,`row-id`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8 COLLATE=utf8_bin;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `phototrack_row_check` (
|
||||||
|
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
|
||||||
|
`table` char(64) NOT NULL,
|
||||||
|
`row-id` int(11) NOT NULL,
|
||||||
|
`checked` timestamp NOT NULL DEFAULT now(),
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
INDEX `row` (`table`,`row-id`),
|
||||||
|
INDEX `checked` (`checked`)
|
||||||
|
) ENGINE=MyISAM DEFAULT CHARSET=utf8 COLLATE=utf8_bin;
|
||||||
|
|
||||||
|
SELECT TRUE
|
274
phototrack/phototrack.php
Normal file
274
phototrack/phototrack.php
Normal file
|
@ -0,0 +1,274 @@
|
||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* Name: Photo Track
|
||||||
|
* Description: Track which photos are actually being used and delete any others
|
||||||
|
* Version: 1.0
|
||||||
|
* Author: Matthew Exon <http://mat.exon.name>
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* List of tables and the fields that are checked:
|
||||||
|
*
|
||||||
|
* contact: photo thumb micro about
|
||||||
|
* fcontact: photo
|
||||||
|
* fsuggest: photo
|
||||||
|
* gcontact: photo about
|
||||||
|
* item: body
|
||||||
|
* item-content: body
|
||||||
|
* mail: from-photo
|
||||||
|
* notify: photo
|
||||||
|
* profile: photo thumb about
|
||||||
|
*/
|
||||||
|
|
||||||
|
use Friendica\Core\Addon;
|
||||||
|
use Friendica\Core\Logger;
|
||||||
|
use Friendica\Object\Image;
|
||||||
|
use Friendica\Database\DBA;
|
||||||
|
use Friendica\Util\Images;
|
||||||
|
use Friendica\Util\DateTimeFormat;
|
||||||
|
use Friendica\DI;
|
||||||
|
|
||||||
|
if (!defined('PHOTOTRACK_DEFAULT_BATCH_SIZE')) {
|
||||||
|
define('PHOTOTRACK_DEFAULT_BATCH_SIZE', 1000);
|
||||||
|
}
|
||||||
|
// Time in *minutes* between searching for photo uses
|
||||||
|
if (!defined('PHOTOTRACK_DEFAULT_SEARCH_INTERVAL')) {
|
||||||
|
define('PHOTOTRACK_DEFAULT_SEARCH_INTERVAL', 10);
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_install() {
|
||||||
|
global $db;
|
||||||
|
|
||||||
|
Addon::registerHook('post_local_end', 'addon/phototrack/phototrack.php', 'phototrack_post_local_end');
|
||||||
|
Addon::registerHook('post_remote_end', 'addon/phototrack/phototrack.php', 'phototrack_post_remote_end');
|
||||||
|
Addon::registerHook('notifier_end', 'addon/phototrack/phototrack.php', 'phototrack_notifier_end');
|
||||||
|
Addon::registerHook('cron', 'addon/phototrack/phototrack.php', 'phototrack_cron');
|
||||||
|
|
||||||
|
if (DI::config()->get('phototrack', 'dbversion') != '0.1') {
|
||||||
|
$schema = file_get_contents(dirname(__file__).'/database.sql');
|
||||||
|
$arr = explode(';', $schema);
|
||||||
|
foreach ($arr as $a) {
|
||||||
|
if (!DBA::e($a)) {
|
||||||
|
Logger::warning('Unable to create database table: ' . DBA::errorMessage());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
DI::config()->set('phototrack', 'dbversion', '0.1');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_uninstall() {
|
||||||
|
Addon::unregisterHook('post_local_end', 'addon/phototrack/phototrack.php', 'phototrack_post_local_end');
|
||||||
|
Addon::unregisterHook('post_remote_end', 'addon/phototrack/phototrack.php', 'phototrack_post_remote_end');
|
||||||
|
Addon::unregisterHook('notifier_end', 'addon/phototrack/phototrack.php', 'phototrack_notifier_end');
|
||||||
|
Addon::unregisterHook('cron', 'addon/phototrack/phototrack.php', 'phototrack_cron');
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_module() {}
|
||||||
|
|
||||||
|
function phototrack_finished_row($table, $id) {
|
||||||
|
$existing = DBA::selectFirst('phototrack_row_check', ['id'], ['table' => $table, 'row-id' => $id]);
|
||||||
|
if (!is_bool($existing)) {
|
||||||
|
DBA::update('phototrack_row_check', ['checked' => DateTimeFormat::utcNow()], ['table' => $table, 'row-id' => $id]);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
DBA::insert('phototrack_row_check', ['table' => $table, 'row-id' => $id, 'checked' => DateTimeFormat::utcNow()]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_photo_use($photo, $table, $field, $id) {
|
||||||
|
Logger::debug('@@@ phototrack_photo_use ' . $photo);
|
||||||
|
foreach (Images::supportedTypes() as $m => $e) {
|
||||||
|
$photo = str_replace(".$e", '', $photo);
|
||||||
|
}
|
||||||
|
if (substr($photo, -2, 1) == '-') {
|
||||||
|
$resolution = intval(substr($photo,-1,1));
|
||||||
|
$photo = substr($photo,0,-2);
|
||||||
|
}
|
||||||
|
if (strlen($photo) != 32) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$r = DBA::selectFirst('photo', ['resource-id'], ['resource-id' => $photo]);
|
||||||
|
if (!DBA::isResult($r)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$rid = $r['resource-id'];
|
||||||
|
$existing = DBA::selectFirst('phototrack_photo_use', ['id'], ['resource-id' => $rid, 'table' => $table, 'field' => $field, 'row-id' => $id]);
|
||||||
|
if (DBA::isResult($existing)) {
|
||||||
|
DBA::update('phototrack_photo_use', ['checked' => DateTimeFormat::utcNow()], ['resource-id' => $rid, 'table' => $table, 'field' => $field, 'row-id' => $id]);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
DBA::insert('phototrack_photo_use', ['resource-id' => $rid, 'table' => $table, 'field' => $field, 'row-id' => $id, 'checked' => DateTimeFormat::utcNow()]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_check_field_url($a, $table, $field, $id, $url) {
|
||||||
|
Logger::info('@@@ phototrack_check_field_url table ' . $table . ' field ' . $field . ' id ' . $id . ' url ' . $url);
|
||||||
|
$baseurl = DI::baseUrl()->get(true);
|
||||||
|
if (strpos($url, $baseurl) === FALSE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$url = substr($url, strlen($baseurl));
|
||||||
|
Logger::info('@@@ phototrack_check_field_url funny url stuff ' . $url . ' base ' . $baseurl);
|
||||||
|
}
|
||||||
|
if (strpos($url, '/photo/') === FALSE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$url = substr($url, strlen('/photo/'));
|
||||||
|
Logger::info('@@@ phototrack_check_field_url more url stuff ' . $url);
|
||||||
|
}
|
||||||
|
if (preg_match('/([0-9a-z]{32})/', $url, $matches)) {
|
||||||
|
$rid = $matches[0];
|
||||||
|
Logger::info('@@@ phototrack_check_field_url rid ' . $rid);
|
||||||
|
phototrack_photo_use($rid, $table, $field, $id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_check_field_bbcode($a, $table, $field, $id, $value) {
|
||||||
|
$baseurl = DI::baseUrl()->get(true);
|
||||||
|
$matches = array();
|
||||||
|
preg_match_all("/\[img(\=([0-9]*)x([0-9]*))?\](.*?)\[\/img\]/ism", $value, $matches);
|
||||||
|
foreach ($matches[4] as $url) {
|
||||||
|
phototrack_check_field_url($a, $table, $field, $id, $url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_post_local_end(&$a, &$item) {
|
||||||
|
phototrack_check_row($a, 'item', $item);
|
||||||
|
phototrack_check_row($a, 'item-content', $item);
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_post_remote_end(&$a, &$item) {
|
||||||
|
phototrack_check_row($a, 'item', $item);
|
||||||
|
phototrack_check_row($a, 'item-content', $item);
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_notifier_end($item) {
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_check_row($a, $table, $row) {
|
||||||
|
switch ($table) {
|
||||||
|
case 'item':
|
||||||
|
$fields = array(
|
||||||
|
'body' => 'bbcode');
|
||||||
|
break;
|
||||||
|
case 'item-content':
|
||||||
|
$fields = array(
|
||||||
|
'body' => 'bbcode');
|
||||||
|
break;
|
||||||
|
case 'contact':
|
||||||
|
$fields = array(
|
||||||
|
'photo' => 'url',
|
||||||
|
'thumb' => 'url',
|
||||||
|
'micro' => 'url',
|
||||||
|
'about' => 'bbcode');
|
||||||
|
break;
|
||||||
|
case 'fcontact':
|
||||||
|
$fields = array(
|
||||||
|
'photo' => 'url');
|
||||||
|
break;
|
||||||
|
case 'fsuggest':
|
||||||
|
$fields = array(
|
||||||
|
'photo' => 'url');
|
||||||
|
break;
|
||||||
|
case 'gcontact':
|
||||||
|
$fields = array(
|
||||||
|
'photo' => 'url',
|
||||||
|
'about' => 'bbcode');
|
||||||
|
break;
|
||||||
|
default: $fields = array(); break;
|
||||||
|
}
|
||||||
|
foreach ($fields as $field => $type) {
|
||||||
|
switch ($type) {
|
||||||
|
case 'bbcode': phototrack_check_field_bbcode($a, $table, $field, $row['id'], $row[$field]); break;
|
||||||
|
case 'url': phototrack_check_field_url($a, $table, $field, $row['id'], $row[$field]); break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
phototrack_finished_row($table, $row['id']);
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_batch_size() {
|
||||||
|
$batch_size = DI::config()->get('phototrack', 'batch_size');
|
||||||
|
if ($batch_size > 0) {
|
||||||
|
return $batch_size;
|
||||||
|
}
|
||||||
|
return PHOTOTRACK_DEFAULT_BATCH_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_search_table($a, $table) {
|
||||||
|
$batch_size = phototrack_batch_size();
|
||||||
|
$rows = DBA::p("SELECT `$table`.* FROM `$table` LEFT OUTER JOIN phototrack_row_check ON ( phototrack_row_check.`table` = '$table' AND phototrack_row_check.`row-id` = `$table`.id ) WHERE ( ( phototrack_row_check.checked IS NULL ) OR ( phototrack_row_check.checked < DATE_SUB(NOW(), INTERVAL 1 MONTH) ) ) ORDER BY phototrack_row_check.checked LIMIT $batch_size");
|
||||||
|
if (DBA::isResult($rows)) {
|
||||||
|
while ($row = DBA::fetch($rows)) {
|
||||||
|
phototrack_check_row($a, $table, $row);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$r = DBA::p("SELECT COUNT(*) FROM `$table` LEFT OUTER JOIN phototrack_row_check ON ( phototrack_row_check.`table` = '$table' AND phototrack_row_check.`row-id` = `$table`.id ) WHERE ( ( phototrack_row_check.checked IS NULL ) OR ( phototrack_row_check.checked < DATE_SUB(NOW(), INTERVAL 1 MONTH) ) )");
|
||||||
|
Logger::info("@@@ phototrack_search_table " . print_r(DBA::fetch($r)));
|
||||||
|
$remaining = DBA::fetch($r)['count'];
|
||||||
|
Logger::info('phototrack: searched ' . DBA::numRows($rows) . ' rows in table ' . $table . ', ' . $remaining . ' still remaining to search');
|
||||||
|
return $remaining;
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_cron_time() {
|
||||||
|
$prev_remaining = DI::config()->get('phototrack', 'remaining_items');
|
||||||
|
if ($prev_remaining > 10 * phototrack_batch_size()) {
|
||||||
|
Logger::debug('phototrack: more than ' . (10 * phototrack_batch_size()) . ' items remaining');
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
$last = DI::config()->get('phototrack', 'last_search');
|
||||||
|
$search_interval = intval(DI::config()->get('phototrack', 'search_interval'));
|
||||||
|
if (!$search_interval) {
|
||||||
|
$search_interval = PHOTOTRACK_DEFAULT_SEARCH_INTERVAL;
|
||||||
|
}
|
||||||
|
if ($last) {
|
||||||
|
$next = $last + ($search_interval * 60);
|
||||||
|
if ($next > time()) {
|
||||||
|
Logger::debug('phototrack: search interval not reached');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_cron($a, $b) {
|
||||||
|
if (!phototrack_cron_time()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
DI::config()->set('phototrack', 'last_search', time());
|
||||||
|
|
||||||
|
$remaining = 0;
|
||||||
|
$remaining += phototrack_search_table($a, 'item');
|
||||||
|
$remaining += phototrack_search_table($a, 'item-content');
|
||||||
|
$remaining += phototrack_search_table($a, 'contact');
|
||||||
|
$remaining += phototrack_search_table($a, 'fcontact');
|
||||||
|
$remaining += phototrack_search_table($a, 'fsuggest');
|
||||||
|
$remaining += phototrack_search_table($a, 'gcontact');
|
||||||
|
|
||||||
|
DI::config()->set('phototrack', 'remaining_items', $remaining);
|
||||||
|
if ($remaining === 0) {
|
||||||
|
phototrack_tidy();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function phototrack_tidy() {
|
||||||
|
$batch_size = phototrack_batch_size();
|
||||||
|
DBA::e('CREATE TABLE IF NOT EXISTS `phototrack-temp` (`resource-id` char(255) not null)');
|
||||||
|
DBA::e('INSERT INTO `phototrack-temp` SELECT DISTINCT(`resource-id`) FROM photo WHERE photo.`created` < DATE_SUB(NOW(), INTERVAL 2 MONTH)');
|
||||||
|
$rows = DBA::p('SELECT `phototrack-temp`.`resource-id` FROM `phototrack-temp` LEFT OUTER JOIN phototrack_photo_use ON (`phototrack-temp`.`resource-id` = phototrack_photo_use.`resource-id`) WHERE phototrack_photo_use.id IS NULL limit ' . /*$batch_size*/1000);
|
||||||
|
if (DBA::isResult($rows)) {
|
||||||
|
foreach ($rows as $row) {
|
||||||
|
Logger::debug('phototrack: remove photo ' . $row['resource-id']);
|
||||||
|
DBA::e('DELETE FROM photo WHERE `resource-id` = "' . $row['resource-id'] . '"');
|
||||||
|
}
|
||||||
|
Logger::info('phototrack_tidy: deleted ' . DBA::numRows($rows) . ' photos');
|
||||||
|
}
|
||||||
|
DBA::e('DROP TABLE `phototrack-temp`');
|
||||||
|
$rows = DBA::p('SELECT id FROM phototrack_photo_use WHERE checked < DATE_SUB(NOW(), INTERVAL 14 DAY)');
|
||||||
|
foreach ($rows as $row) {
|
||||||
|
DBA::e( 'DELETE FROM phototrack_photo_use WHERE id = ' . $row['id']);
|
||||||
|
}
|
||||||
|
Logger::info('phototrack_tidy: deleted ' . DBA::numRows($rows) . ' phototrack_photo_use rows');
|
||||||
|
}
|
11
publicise/publicise.php
Normal file
11
publicise/publicise.php
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
"SELECT `uid` FROM `contact` WHERE `id` = %d AND `reason` = 'publicise'", intval($item['contact-id']));
|
||||||
|
if (!$r1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Logger::debug('Publicise: moving to wall: ' . $item['uid'] . ' ' . $item['contact-id'] . ' ' . $item['uri']);
|
||||||
|
$item['type'] = 'wall';
|
||||||
|
$item['wall'] = 1;
|
||||||
|
$item['private'] = 0;
|
||||||
|
}
|
||||||
|
|
39
publicise/templates/admin.tpl
Normal file
39
publicise/templates/admin.tpl
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
{{*
|
||||||
|
* AUTOMATICALLY GENERATED TEMPLATE
|
||||||
|
* DO NOT EDIT THIS FILE, CHANGES WILL BE OVERWRITTEN
|
||||||
|
*
|
||||||
|
*}}
|
||||||
|
<form method="post">
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>{{$feed_t}}</th>
|
||||||
|
<th>{{$publicised_t}}</th>
|
||||||
|
<th>{{$comments_t}}</th>
|
||||||
|
<th>{{$expire_t}}</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{{foreach $feeds as $f}}
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
<a href="{{$f.url}}">
|
||||||
|
<img style="vertical-align:middle" src='{{$f.micro}}'>
|
||||||
|
<span style="margin-left:1em">{{$f.name}}</span>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
{{include file="field_yesno.tpl" field=$f.enabled}}
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
{{include file="field_yesno.tpl" field=$f.comments}}
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<input name="publicise-expire-{{$f.id}}" value="{{$f.expire}}">
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{{/foreach}}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<input type="submit" size="70" value="{{$submit_t}}">
|
||||||
|
</form>
|
42
retriever/database.sql
Normal file
42
retriever/database.sql
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
CREATE TABLE IF NOT EXISTS `retriever_rule` (
|
||||||
|
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
|
||||||
|
`uid` int(11) NOT NULL,
|
||||||
|
`contact-id` int(11) NOT NULL,
|
||||||
|
`data` mediumtext NULL DEFAULT NULL,
|
||||||
|
PRIMARY KEY (`id`),
|
||||||
|
KEY `uid` (`uid`),
|
||||||
|
KEY `contact-id` (`contact-id`)
|
||||||
|
) DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `retriever_item` (
|
||||||
|
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
|
||||||
|
`item-uri` varbinary(255) NOT NULL,
|
||||||
|
`item-uid` int(10) unsigned NOT NULL DEFAULT '0',
|
||||||
|
`contact-id` int(10) unsigned NOT NULL DEFAULT '0',
|
||||||
|
`resource` int(11) NOT NULL,
|
||||||
|
`finished` tinyint(1) unsigned NOT NULL DEFAULT '0',
|
||||||
|
KEY `resource` (`resource`),
|
||||||
|
KEY `finished` (`finished`),
|
||||||
|
KEY `item-uid` (`item-uid`),
|
||||||
|
KEY `all` (`item-uri`, `item-uid`, `contact-id`),
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci;
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS `retriever_resource` (
|
||||||
|
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
|
||||||
|
`item-uid` int(10) unsigned NOT NULL DEFAULT '0',
|
||||||
|
`contact-id` int(10) unsigned NOT NULL DEFAULT '0',
|
||||||
|
`type` char(255) NULL DEFAULT NULL,
|
||||||
|
`binary` int(1) NOT NULL DEFAULT 0,
|
||||||
|
`url` varbinary(700) NOT NULL,
|
||||||
|
`created` timestamp NOT NULL DEFAULT now(),
|
||||||
|
`completed` timestamp NULL DEFAULT NULL,
|
||||||
|
`last-try` timestamp NULL DEFAULT NULL,
|
||||||
|
`num-tries` int(11) NOT NULL DEFAULT 0,
|
||||||
|
`data` mediumblob NULL DEFAULT NULL,
|
||||||
|
`http-code` smallint(1) unsigned NULL DEFAULT NULL,
|
||||||
|
`redirect-url` varbinary(700) NOT NULL,
|
||||||
|
KEY `url` (`url`),
|
||||||
|
KEY `completed` (`completed`),
|
||||||
|
PRIMARY KEY (`id`)
|
||||||
|
) DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci;
|
1058
retriever/retriever.php
Normal file
1058
retriever/retriever.php
Normal file
File diff suppressed because it is too large
Load diff
9
retriever/templates/admin.tpl
Normal file
9
retriever/templates/admin.tpl
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
{{*
|
||||||
|
* AUTOMATICALLY GENERATED TEMPLATE
|
||||||
|
* DO NOT EDIT THIS FILE, CHANGES WILL BE OVERWRITTEN
|
||||||
|
*
|
||||||
|
*}}
|
||||||
|
{{include file="field_input.tpl" field=$downloads_per_cron}}
|
||||||
|
{{include file="field_checkbox.tpl" field=$allow_images}}
|
||||||
|
<div class="submit"><input type="submit" name="page_site" value="{{$submit}}"></div>
|
||||||
|
|
24
retriever/templates/extract.tpl
Normal file
24
retriever/templates/extract.tpl
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
|
||||||
|
<xsl:output method="html" indent="yes" version="4.0"/>
|
||||||
|
|
||||||
|
<xsl:template match="text()"/>
|
||||||
|
{{function clause_xpath}}{{if !$clause.attribute}}{{$clause.element}}{{elseif $clause.attribute == 'class'}}{{$clause.element}}[contains(concat(' ', normalize-space(@class), ' '), '{{$clause.value}}')]{{else}}{{$clause.element}}[@{{$clause.attribute}}='{{$clause.value}}']{{/if}}{{/function}}
|
||||||
|
{{foreach $spec.include as $clause}}
|
||||||
|
|
||||||
|
<xsl:template match="{{clause_xpath clause=$clause}}">
|
||||||
|
<xsl:copy>
|
||||||
|
<xsl:apply-templates select="node()|@*" mode="remove"/>
|
||||||
|
</xsl:copy>
|
||||||
|
</xsl:template>{{/foreach}}
|
||||||
|
{{foreach $spec.exclude as $clause}}
|
||||||
|
|
||||||
|
<xsl:template match="{{clause_xpath clause=$clause}}" mode="remove"/>{{/foreach}}
|
||||||
|
|
||||||
|
<xsl:template match="node()|@*" mode="remove">
|
||||||
|
<xsl:copy>
|
||||||
|
<xsl:apply-templates select="node()|@*" mode="remove"/>
|
||||||
|
</xsl:copy>
|
||||||
|
</xsl:template>
|
||||||
|
|
||||||
|
</xsl:stylesheet>
|
26
retriever/templates/fix-urls.tpl
Normal file
26
retriever/templates/fix-urls.tpl
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
|
||||||
|
<!-- attempt to replace relative URLs with absolute URLs -->
|
||||||
|
<!-- http://stackoverflow.com/questions/3824631/replace-href-value-in-anchor-tags-of-html-using-xslt -->
|
||||||
|
|
||||||
|
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
|
||||||
|
<xsl:output method="html" indent="yes" version="4.0"/>
|
||||||
|
|
||||||
|
<xsl:template match="node()|@*">
|
||||||
|
<xsl:copy>
|
||||||
|
<xsl:apply-templates select="node()|@*"/>
|
||||||
|
</xsl:copy>
|
||||||
|
</xsl:template>
|
||||||
|
|
||||||
|
<xsl:template match="*/@src[starts-with(.,'.')]">
|
||||||
|
<xsl:attribute name="src">
|
||||||
|
<xsl:value-of select="concat('{{$dirurl}}',.)"/>
|
||||||
|
</xsl:attribute>
|
||||||
|
</xsl:template>
|
||||||
|
<xsl:template match="*/@src[starts-with(.,'/')]">
|
||||||
|
<xsl:attribute name="src">
|
||||||
|
<xsl:value-of select="concat('{{$rooturl}}',.)"/>
|
||||||
|
</xsl:attribute>
|
||||||
|
</xsl:template>
|
||||||
|
|
||||||
|
</xsl:stylesheet>
|
163
retriever/templates/help.tpl
Normal file
163
retriever/templates/help.tpl
Normal file
|
@ -0,0 +1,163 @@
|
||||||
|
<h2>Retriever Plugin Help</h2>
|
||||||
|
<p>
|
||||||
|
This plugin replaces the short excerpts you normally get in RSS feeds
|
||||||
|
with the full content of the article from the source website. You
|
||||||
|
specify which part of the page you're interested in with a set of
|
||||||
|
rules. When each item arrives, the plugin downloads the full page
|
||||||
|
from the website, extracts content using the rules, and replaces the
|
||||||
|
original article.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
There's a few reasons you may want to do this. The source website
|
||||||
|
might be slow or overloaded. The source website might be
|
||||||
|
untrustworthy, in which case using Friendica to scrub the HTML is a
|
||||||
|
good idea. You might be on a LAN that blacklists certain websites.
|
||||||
|
It also works neatly with the mailstream plugin, allowing you to read
|
||||||
|
a news stream comfortably without needing continuous Internet
|
||||||
|
connectivity.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
However, setting up retriever can be quite tricky since it depends on
|
||||||
|
the internal design of the website. That was designed to make life
|
||||||
|
easy for the website's developers, not for you. You'll need to have
|
||||||
|
some familiarity with HTML, and be willing to adapt when the website
|
||||||
|
suddenly changes everything without notice.
|
||||||
|
</p>
|
||||||
|
<h3>Configuring Retriever for a feed</h3>
|
||||||
|
<p>
|
||||||
|
To set up retriever for an RSS feed, go to the "Contacts" page and
|
||||||
|
find your feed. Then click on the drop-down menu on the contact.
|
||||||
|
Select "Retriever" to get to the retriever configuration.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
The "Include" configuration section specifies parts of the page to
|
||||||
|
include in the article. Each row has three components:
|
||||||
|
</p>
|
||||||
|
<ul>
|
||||||
|
<li>An HTML tag (e.g. "div", "span", "p")</li>
|
||||||
|
<li>An attribute (usually "class" or "id")</li>
|
||||||
|
<li>A value for the attribute</li>
|
||||||
|
</ul>
|
||||||
|
<p>
|
||||||
|
A simple case is when the article is wrapped in a "div" element:
|
||||||
|
</p>
|
||||||
|
<pre>
|
||||||
|
...
|
||||||
|
<div class="ArticleWrapper">
|
||||||
|
<h2>Man Bites Dog</h2>
|
||||||
|
<img src="mbd.jpg">
|
||||||
|
<p>
|
||||||
|
Residents of the sleepy community of Nowheresville were
|
||||||
|
shocked yesterday by the sight of creepy local weirdo Jim
|
||||||
|
McOddman assaulting innocent local dog Snufflekins with his
|
||||||
|
false teeth.
|
||||||
|
</p>
|
||||||
|
...
|
||||||
|
</div>
|
||||||
|
...
|
||||||
|
</pre>
|
||||||
|
<p>
|
||||||
|
You then specify the tag "div", attribute "class", and value
|
||||||
|
"ArticleWrapper". Everything else in the page, such as navigation
|
||||||
|
panels and menus and footers and so on, will be discarded. If there
|
||||||
|
is more than one section of the page you want to include, specify each
|
||||||
|
one on a separate row. If the matching section contains some sections
|
||||||
|
you want to remove, specify those in the "Exclude" section in the same
|
||||||
|
way.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
Once you've got a configuration that you think will work, you can try
|
||||||
|
it out on some existing articles. Type a number into the
|
||||||
|
"Retrospectively Apply" box and click "Submit". After a while
|
||||||
|
(exactly how long depends on your system's cron configuration) the new
|
||||||
|
articles should be available.
|
||||||
|
</p>
|
||||||
|
<h3>Techniques</h3>
|
||||||
|
<p>
|
||||||
|
You can leave the attribute and value blank to include all the
|
||||||
|
corresponding elements with the specified tag name. You can also use
|
||||||
|
a tag name of just an asterisk ("*"), which will match any element type with the
|
||||||
|
specified attribute regardless of the tag.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
Note that the "class" attribute is a special case. Many web page
|
||||||
|
templates will put multiple different classes in the same element,
|
||||||
|
separated by spaces. If you specify an attribute of "class" it will
|
||||||
|
match an element if any of its classes matches the specified value.
|
||||||
|
For example:
|
||||||
|
</p>
|
||||||
|
<pre>
|
||||||
|
<div class="article breaking-news">
|
||||||
|
</pre>
|
||||||
|
<p>
|
||||||
|
In this case you can specify a value of "article", or "breaking-news".
|
||||||
|
You can also specify "article breaking-news", but that won't match if
|
||||||
|
the website suddenly changes to "breaking-news article", so that's not
|
||||||
|
recommended.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
One useful trick you can try is using the website's "print" pages.
|
||||||
|
Many news sites have print versions of all their articles. These are
|
||||||
|
usually drastically simplified compared to the live website page.
|
||||||
|
Sometimes this is a good way to get the whole article when it's
|
||||||
|
normally split across multiple pages.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
Hopefully the URL for the print page is a predictable variant of the
|
||||||
|
normal article URL. For example, an article URL like:
|
||||||
|
</p>
|
||||||
|
<pre>
|
||||||
|
http://www.newssite.com/article-8636.html
|
||||||
|
</pre>
|
||||||
|
<p>
|
||||||
|
...might have a print version at:
|
||||||
|
</p>
|
||||||
|
<pre>
|
||||||
|
http://www.newssite.com/print/article-8636.html
|
||||||
|
</pre>
|
||||||
|
<p>
|
||||||
|
To change the URL used to retrieve the page, use the "URL Pattern" and
|
||||||
|
"URL Replace" fields. The pattern is a regular expression matching
|
||||||
|
part of the URL to replace. In this case, you might use a pattern of
|
||||||
|
"/article" and a replace string of "/print/article". A common pattern
|
||||||
|
is simply a dollar sign ("$"), used to add the replace string to the end of the URL.
|
||||||
|
</p>
|
||||||
|
<h3>Background Processing</h3>
|
||||||
|
<p>
|
||||||
|
Note that retrieving and processing the articles can take some time,
|
||||||
|
so it's done in the background. Incoming articles will be marked as
|
||||||
|
invisible while they're in the process of being downloaded. If a URL
|
||||||
|
fails, the plugin will keep trying at progressively longer intervals
|
||||||
|
for up to a month, in case the website is temporarily overloaded or
|
||||||
|
the network is down.
|
||||||
|
</p>
|
||||||
|
{{if $allow_images}}
|
||||||
|
<h3>Retrieving Images</h3>
|
||||||
|
<p>
|
||||||
|
Retriever can also optionally download images and store them in the
|
||||||
|
local Friendica instance. Just check the "Download Images" box. You
|
||||||
|
can also download images in every item from your network, whether it's
|
||||||
|
an RSS feed or not. Go to the "Settings" page and
|
||||||
|
click <a href="$config">"Plugin settings"</a>. Then check the "All
|
||||||
|
Photos" box in the "Retriever Settings" section and click "Submit".
|
||||||
|
</p>
|
||||||
|
{{/if}}
|
||||||
|
<h2>Configure Feeds:</h2>
|
||||||
|
<div>
|
||||||
|
{{foreach $feeds as $feed}}
|
||||||
|
<div class="contact-entry-wrapper" id="contact-entry-wrapper-{{$feed.id}}">
|
||||||
|
<a href="{{$feed.url}} title="{{$feed.img_hover}}">
|
||||||
|
<div class="contact-entry-photo-wrapper">
|
||||||
|
<div class="contact-entry-photo mframe" id="contact-entry-photo-{{$feed.id}}">
|
||||||
|
<img src="{{$feed.thumb}}" {{$feed.sparkle}} alt="{{$feed.name}}"/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="contact-entry-desc">
|
||||||
|
<div class="contact-entry-name" id="contact-entry-name-{{$feed.id}}">
|
||||||
|
{{$feed.name}}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
{{/foreach}}
|
||||||
|
</div>
|
154
retriever/templates/rule-config.tpl
Normal file
154
retriever/templates/rule-config.tpl
Normal file
|
@ -0,0 +1,154 @@
|
||||||
|
<div class="settings-block">
|
||||||
|
<script language="javascript">
|
||||||
|
function retriever_add_row(id)
|
||||||
|
{
|
||||||
|
var tbody = document.getElementById(id);
|
||||||
|
var last = tbody.rows[tbody.childElementCount - 1];
|
||||||
|
var count = +last.id.replace(id + '-', '');
|
||||||
|
count++;
|
||||||
|
var row = document.createElement('tr');
|
||||||
|
row.id = id + '-' + count;
|
||||||
|
var cell1 = document.createElement('td');
|
||||||
|
var inptag = document.createElement('input');
|
||||||
|
inptag.name = row.id + '-element';
|
||||||
|
cell1.appendChild(inptag);
|
||||||
|
row.appendChild(cell1);
|
||||||
|
var cell2 = document.createElement('td');
|
||||||
|
var inpatt = document.createElement('input');
|
||||||
|
inpatt.name = row.id + '-attribute';
|
||||||
|
cell2.appendChild(inpatt);
|
||||||
|
row.appendChild(cell2);
|
||||||
|
var cell3 = document.createElement('td');
|
||||||
|
var inpval = document.createElement('input');
|
||||||
|
inpval.name = row.id + '-value';
|
||||||
|
cell3.appendChild(inpval);
|
||||||
|
row.appendChild(cell3);
|
||||||
|
var cell4 = document.createElement('td');
|
||||||
|
var butrem = document.createElement('input');
|
||||||
|
butrem.id = row.id + '-rem';
|
||||||
|
butrem.type = 'button';
|
||||||
|
butrem.onclick = function(){retriever_remove_row(id, count)};
|
||||||
|
butrem.value = '{{$remove_t}}';
|
||||||
|
cell4.appendChild(butrem);
|
||||||
|
row.appendChild(cell4);
|
||||||
|
tbody.appendChild(row);
|
||||||
|
}
|
||||||
|
|
||||||
|
function retriever_remove_row(id, number)
|
||||||
|
{
|
||||||
|
var tbody = document.getElementById(id);
|
||||||
|
var row = document.getElementById(id + '-' + number);
|
||||||
|
tbody.removeChild(row);
|
||||||
|
}
|
||||||
|
|
||||||
|
function retriever_toggle_url_block()
|
||||||
|
{
|
||||||
|
var pattern = document.querySelector("#id_retriever_pattern").parentNode;
|
||||||
|
if (document.querySelector("#id_retriever_modurl").checked) {
|
||||||
|
pattern.style.display = "block";
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
pattern.style.display = "none";
|
||||||
|
}
|
||||||
|
|
||||||
|
var replace = document.querySelector("#id_retriever_replace").parentNode;
|
||||||
|
if (document.querySelector("#id_retriever_modurl").checked) {
|
||||||
|
replace.style.display = "block";
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
replace.style.display = "none";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function retriever_toggle_cookiedata_block()
|
||||||
|
{
|
||||||
|
var div = document.querySelector("#id_retriever_cookiedata").parentNode;
|
||||||
|
if (document.querySelector("#id_retriever_storecookies").checked) {
|
||||||
|
div.style.display = "block";
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
div.style.display = "none";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
|
retriever_toggle_url_block();
|
||||||
|
document.querySelector("#id_retriever_modurl").addEventListener('change', retriever_toggle_url_block, false);
|
||||||
|
retriever_toggle_cookiedata_block();
|
||||||
|
document.querySelector("#id_retriever_storecookies").addEventListener('change', retriever_toggle_cookiedata_block, false);
|
||||||
|
}, false);
|
||||||
|
</script>
|
||||||
|
<h2>{{$title}}</h2>
|
||||||
|
<p><a href="{{$help}}">{{$help_t}}</a></p>
|
||||||
|
<form method="post">
|
||||||
|
<input type="hidden" name="id" value="{{$id}}">
|
||||||
|
{{include file="field_checkbox.tpl" field=$enable}}
|
||||||
|
<h3>{{$include_t}}:</h3>
|
||||||
|
<div>
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr><th>{{$tag_t}}</th><th>{{$attribute_t}}</th><th>{{$value_t}}</th></tr>
|
||||||
|
</thead>
|
||||||
|
<tbody id="retriever-include">
|
||||||
|
{{if $include}}
|
||||||
|
{{foreach $include as $k=>$m}}
|
||||||
|
<tr id="retriever-include-{{$k}}">
|
||||||
|
<td><input name="retriever-include-{{$k}}-element" value="{{$m.element}}"></td>
|
||||||
|
<td><input name="retriever-include-{{$k}}-attribute" value="{{$m.attribute}}"></td>
|
||||||
|
<td><input name="retriever-include-{{$k}}-value" value="{{$m.value}}"></td>
|
||||||
|
<td><input id="retrieve-include-{{$k}}-rem" type="button" onclick="retriever_remove_row('retriever-include', {{$k}})" value="{{$remove_t}}"></td>
|
||||||
|
</tr>
|
||||||
|
{{/foreach}}
|
||||||
|
{{else}}
|
||||||
|
<tr id="retriever-include-0">
|
||||||
|
<td><input name="retriever-include-0-element"></td>
|
||||||
|
<td><input name="retriever-include-0-attribute"></td>
|
||||||
|
<td><input name="retriever-include-0-value"></td>
|
||||||
|
<td><input id="retrieve-include-0-rem" type="button" onclick="retriever_remove_row('retriever-include', 0)" value="{{$remove_t}}"></td>
|
||||||
|
</tr>
|
||||||
|
{{/if}}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<input type="button" onclick="retriever_add_row('retriever-include')" value="{{$add_t}}">
|
||||||
|
</div>
|
||||||
|
<h3>{{$exclude_t}}:</h3>
|
||||||
|
<div>
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr><th>{{$tag_t}}</th><th>{{$attribute_t}}</th><th>{{$value_t}}</th></tr>
|
||||||
|
</thead>
|
||||||
|
<tbody id="retriever-exclude">
|
||||||
|
{{if $exclude}}
|
||||||
|
{{foreach $exclude as $k=>$r}}
|
||||||
|
<tr id="retriever-exclude-{{$k}}">
|
||||||
|
<td><input name="retriever-exclude-{{$k}}-element" value="{{$r.element}}"></td>
|
||||||
|
<td><input name="retriever-exclude-{{$k}}-attribute" value="{{$r.attribute}}"></td>
|
||||||
|
<td><input name="retriever-exclude-{{$k}}-value" value="{{$r.value}}"></td>
|
||||||
|
<td><input id="retrieve-exclude-{{$k}}-rem" type="button" onclick="retriever_remove_row('retriever-exclude', {{$k}})" value="{{$remove_t}}"></td>
|
||||||
|
</tr>
|
||||||
|
{{/foreach}}
|
||||||
|
{{else}}
|
||||||
|
<tr id="retriever-exclude-0">
|
||||||
|
<td><input name="retriever-exclude-0-element"></td>
|
||||||
|
<td><input name="retriever-exclude-0-attribute"></td>
|
||||||
|
<td><input name="retriever-exclude-0-value"></td>
|
||||||
|
<td><input id="retrieve-exclude-0-rem" type="button" onclick="retriever_remove_row('retriever-exclude', 0)" value="{{$remove_t}}"></td>
|
||||||
|
</tr>
|
||||||
|
{{/if}}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<input type="button" onclick="retriever_add_row('retriever-exclude')" value="{{$add_t}}">
|
||||||
|
</div>
|
||||||
|
{{include file="field_checkbox.tpl" field=$modurl}}
|
||||||
|
{{include file="field_input.tpl" field=$pattern}}
|
||||||
|
{{include file="field_input.tpl" field=$replace}}
|
||||||
|
{{if $allow_images}}
|
||||||
|
{{include file="field_checkbox.tpl" field=$images}}
|
||||||
|
{{/if}}
|
||||||
|
{{include file="field_textarea.tpl" field=$customxslt}}
|
||||||
|
{{include file="field_checkbox.tpl" field=$storecookies}}
|
||||||
|
{{include file="field_textarea.tpl" field=$cookiedata}}
|
||||||
|
{{include file="field_input.tpl" field=$retrospective}}
|
||||||
|
<input type="submit" size="70" value="{{$submit_t}}">
|
||||||
|
</form>
|
||||||
|
</div>
|
16
retriever/templates/settings.tpl
Normal file
16
retriever/templates/settings.tpl
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
<span id="settings_retriever_inflated" class="settings-block fakelink" style="display: block;" onclick="openClose('settings_retriever_expanded'); openClose('settings_retriever_inflated');">
|
||||||
|
<h3>{{$title}}</h3>
|
||||||
|
</span>
|
||||||
|
<div id="settings_retriever_expanded" class="settings-block" style="display: none;">
|
||||||
|
<span class="fakelink" onclick="openClose('settings_retriever_expanded'); openClose('settings_retriever_inflated');">
|
||||||
|
<h3>{{$title}}</h3>
|
||||||
|
</span>
|
||||||
|
<p>
|
||||||
|
<a href="{{$help}}">Get Help</a>
|
||||||
|
</p>
|
||||||
|
{{if $allow_images}}
|
||||||
|
{{include file="field_checkbox.tpl" field=$allphotos}}
|
||||||
|
{{/if}}
|
||||||
|
{{include file="field_checkbox.tpl" field=$oembed}}
|
||||||
|
<input type="submit" value="{{$submit}}">
|
||||||
|
</div>
|
|
@ -1363,7 +1363,7 @@ function twitter_fetchtimeline(int $uid): void
|
||||||
|
|
||||||
Logger::info('Posting mirror post', ['twitter-id' => $post->id_str, 'uid' => $uid]);
|
Logger::info('Posting mirror post', ['twitter-id' => $post->id_str, 'uid' => $uid]);
|
||||||
|
|
||||||
Post\Delayed::add($mirrorpost['extid'], $mirrorpost, Worker::PRIORITY_MEDIUM, Post\Delayed::PREPARED);
|
Post\Delayed::add($mirrorpost['extid'], $mirrorpost, Worker::PRIORITY_MEDIUM, Post\Delayed::UNPREPARED);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DI::pConfig()->set($uid, 'twitter', 'lastid', $lastid);
|
DI::pConfig()->set($uid, 'twitter', 'lastid', $lastid);
|
||||||
|
|
Loading…
Reference in a new issue