[NodeInfo] New endpoint and formula for computing active users

Seriously improved documentation
Now NodeInfo 2.0 is available at /api/nodeinfo/2.0.json
For active users we now also consider favourites and recently created accounts
Some further minor bug fixes and full review of the implementation
This commit is contained in:
Diogo Cordeiro 2019-08-12 04:45:25 +01:00
parent c0ac7f0ac8
commit e4bdb21a54
7 changed files with 519 additions and 179 deletions

View File

@ -2657,7 +2657,7 @@ function common_strip_html($html, $trim=true, $save_whitespace=false)
* @param string|bool $size * @param string|bool $size
* @return int the php.ini upload limit in machine-readable format * @return int the php.ini upload limit in machine-readable format
*/ */
function _common_size_str_to_int($size) : int function _common_size_str_to_int($size): int
{ {
// `memory_limit` can be -1 and `post_max_size` can be 0 // `memory_limit` can be -1 and `post_max_size` can be 0
// for unlimited. Consistency. // for unlimited. Consistency.
@ -2692,7 +2692,7 @@ function _common_size_str_to_int($size) : int
* *
* @return int * @return int
*/ */
function common_get_preferred_php_upload_limit() : int { function common_get_preferred_php_upload_limit(): int {
return min(_common_size_str_to_int(ini_get('post_max_size')), return min(_common_size_str_to_int(ini_get('post_max_size')),
_common_size_str_to_int(ini_get('upload_max_filesize')), _common_size_str_to_int(ini_get('upload_max_filesize')),
_common_size_str_to_int(ini_get('memory_limit'))); _common_size_str_to_int(ini_get('memory_limit')));

View File

@ -1,20 +1,52 @@
<?php <?php
// This file is part of GNU social - https://www.gnu.org/software/social
//
// GNU social 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.
//
// GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>.
if (!defined('GNUSOCIAL')) { /**
exit(1); * Plugin that presents basic instance information using the [NodeInfo standard](http://nodeinfo.diaspora.software/).
} *
* @package NodeInfo
* @author Stéphane Bérubé <chimo@chromic.org>
* @author Diogo Cordeiro <diogo@fc.up.pt>
* @copyright 2018-2019 Free Software Foundation, Inc http://www.fsf.org
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*/
defined('GNUSOCIAL') || die();
/**
* Controls cache and routes
*
* @copyright 2018-2019 Free Software Foundation, Inc http://www.fsf.org
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*/
class NodeinfoPlugin extends Plugin class NodeinfoPlugin extends Plugin
{ {
const PLUGIN_VERSION = '1.0.2'; const PLUGIN_VERSION = '2.0.0';
public function onRouterInitialized($m) public function onRouterInitialized($m): bool
{ {
$m->connect('.well-known/nodeinfo', $m->connect(
['action' => 'nodeinfojrd']); '.well-known/nodeinfo',
['action' => 'nodeinfojrd']
);
$m->connect('main/nodeinfo/2.0', $m->connect(
['action' => 'nodeinfo_2_0']); 'api/nodeinfo/2.0.json',
['action' => 'nodeinfo_2_0']
);
return true; return true;
} }
@ -22,9 +54,10 @@ class NodeinfoPlugin extends Plugin
/** /**
* Make sure necessary tables are filled out. * Make sure necessary tables are filled out.
* *
* @return boolean hook true * @return bool hook true
* @author Diogo Cordeiro <diogo@fc.up.pt>
*/ */
public function onCheckSchema() public function onCheckSchema(): bool
{ {
// Ensure schema // Ensure schema
$schema = Schema::get(); $schema = Schema::get();
@ -55,10 +88,11 @@ class NodeinfoPlugin extends Plugin
/** /**
* Increment notices/replies counter * Increment notices/replies counter
* *
* @return boolean hook flag * @param Notice $notice
* @return bool hook flag
* @author Diogo Cordeiro <diogo@fc.up.pt> * @author Diogo Cordeiro <diogo@fc.up.pt>
*/ */
public function onStartNoticeDistribute($notice) public function onStartNoticeDistribute(Notice $notice): bool
{ {
assert($notice->id > 0); // Ignore if not a valid notice assert($notice->id > 0); // Ignore if not a valid notice
@ -68,16 +102,18 @@ class NodeinfoPlugin extends Plugin
return true; return true;
} }
// Ignore for activity/non-post-verb notices // Ignore for activity/non-(post/share)-verb notices
if (method_exists('ActivityUtils', 'compareVerbs')) { if (method_exists('ActivityUtils', 'compareVerbs')) {
$is_post_verb = ActivityUtils::compareVerbs( $is_valid_verb = ActivityUtils::compareVerbs(
$notice->verb, $notice->verb,
[ActivityVerb::POST] [ActivityVerb::POST,
ActivityVerb::SHARE]
); );
} else { } else {
$is_post_verb = ($notice->verb == ActivityVerb::POST ? true : false); $is_valid_verb = ($notice->verb == ActivityVerb::POST ||
$notice->verb == ActivityVerb::SHARE);
} }
if ($notice->source == 'activity' || !$is_post_verb) { if ($notice->source == 'activity' || !$is_valid_verb) {
return true; return true;
} }
@ -105,10 +141,13 @@ class NodeinfoPlugin extends Plugin
/** /**
* Decrement notices/replies counter * Decrement notices/replies counter
* *
* @return boolean hook flag * @param User $user
* @param Notice $notice
* @return bool hook flag
* @throws UserNoProfileException
* @author Diogo Cordeiro <diogo@fc.up.pt> * @author Diogo Cordeiro <diogo@fc.up.pt>
*/ */
public function onStartDeleteOwnNotice($user, $notice) public function onStartDeleteOwnNotice(User $user, Notice $notice): bool
{ {
$profile = $user->getProfile(); $profile = $user->getProfile();
@ -133,10 +172,10 @@ class NodeinfoPlugin extends Plugin
/** /**
* Increment users counter * Increment users counter
* *
* @return boolean hook flag * @return bool hook flag
* @author Diogo Cordeiro <diogo@fc.up.pt> * @author Diogo Cordeiro <diogo@fc.up.pt>
*/ */
public function onEndRegistrationTry() public function onEndRegistrationTry(): bool
{ {
$us = Usage_stats::getKV('type', 'users'); $us = Usage_stats::getKV('type', 'users');
$us->count += 1; $us->count += 1;
@ -147,10 +186,10 @@ class NodeinfoPlugin extends Plugin
/** /**
* Decrement users counter * Decrement users counter
* *
* @return boolean hook flag * @return bool hook flag
* @author Diogo Cordeiro <diogo@fc.up.pt> * @author Diogo Cordeiro <diogo@fc.up.pt>
*/ */
public function onEndDeleteUser() public function onEndDeleteUser(): bool
{ {
$us = Usage_stats::getKV('type', 'users'); $us = Usage_stats::getKV('type', 'users');
$us->count -= 1; $us->count -= 1;
@ -158,23 +197,38 @@ class NodeinfoPlugin extends Plugin
return true; return true;
} }
/**
public function onPluginVersion(array &$versions) * Plugin version information
*
* @param array $versions
* @return bool hook true
* @throws Exception
*/
public function onPluginVersion(array &$versions): bool
{ {
$versions[] = ['name' => 'Nodeinfo', $versions[] = [
'name' => 'Nodeinfo',
'version' => self::PLUGIN_VERSION, 'version' => self::PLUGIN_VERSION,
'author' => 'chimo', 'author' => 'Stéphane Bérubé, Diogo Cordeiro',
'homepage' => 'https://github.com/chimo/gs-nodeinfo', 'homepage' => 'https://code.chromic.org/chimo/gs-nodeinfo',
'description' => _m('Plugin that presents basic instance information using the NodeInfo standard.')]; 'description' => _m('Plugin that presents basic instance information using the NodeInfo standard.')
];
return true; return true;
} }
public function onEndUpgrade() /**
* Cache was added in a newer version of the plugin, this ensures we fix cached values on upgrade
*
* @return bool hook flag
* @author Diogo Cordeiro <diogo@fc.up.pt>
*/
public function onEndUpgrade(): bool
{ {
$users = new Usage_stats(); $users = new Usage_stats();
if ($users->getUserCount() == 0) { if ($users->getUserCount() == 0) {
define('NODEINFO_UPGRADE', true); define('NODEINFO_UPGRADE', true);
require_once __DIR__ . DIRECTORY_SEPARATOR . 'scripts' . DIRECTORY_SEPARATOR . 'fix_stats.php'; require_once __DIR__ . DIRECTORY_SEPARATOR . 'scripts' . DIRECTORY_SEPARATOR . 'fix_stats.php';
} }
return true;
} }
} }

View File

@ -1,7 +1,8 @@
# Nodeinfo plugin for GNU social # Nodeinfo support for GNU social
Plugin that presents basic instance information using the [NodeInfo standard](http://nodeinfo.diaspora.software/). Plugin that presents basic instance information using the [NodeInfo standard](http://nodeinfo.diaspora.software/).
At the moment, the information is presented at the "/main/nodeinfo/2.0" endpoint. The information is presented at the "/nodeinfo/2.0.json" endpoint.
Other tools can then scrape that information and present it in various ways. For example: [https://fediverse.network/](https://fediverse.network/) Other tools can then scrape that information and present it in various ways.
For example: [https://fediverse.network/](https://fediverse.network/)

View File

@ -1,43 +1,78 @@
<?php <?php
// This file is part of GNU social - https://www.gnu.org/software/social
//
// GNU social 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.
//
// GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>.
if (!defined('GNUSOCIAL')) { /**
exit(1); * The information is presented at the "api/nodeinfo/2.0.json" endpoint.
} *
* @package NodeInfo
* @author Stéphane Bérubé <chimo@chromic.org>
* @author Diogo Cordeiro <diogo@fc.up.pt>
* @copyright 2018-2019 Free Software Foundation, Inc http://www.fsf.org
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*/
class Nodeinfo_2_0Action extends ApiAction defined('GNUSOCIAL') || die();
/**
* NodeInfo 2.0
*
* @copyright 2018-2019 Free Software Foundation, Inc http://www.fsf.org
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*/
class Nodeinfo_2_0Action extends Action
{ {
private $plugins; private $plugins;
protected function handle() protected function handle(): void
{ {
parent::handle(); parent::handle();
header('Access-Control-Allow-Origin: *');
$this->plugins = $this->getActivePluginList(); $this->plugins = $this->getActivePluginList();
$this->showNodeInfo(); $this->showNodeInfo();
} }
public function getActivePluginList() /**
* Most functionality depends on the active plugins, this gives us enough information concerning that
*
* @return array
* @author Stéphane Bérubé <chimo@chromic.org>
* @author Diogo Cordeiro <diogo@fc.up.pt>
*/
public function getActivePluginList(): array
{ {
$pluginversions = array(); $plugin_version = [];
$plugins = array(); $plugins = [];
Event::handle('PluginVersion', array(&$pluginversions)); Event::handle('PluginVersion', [&$plugin_version]);
foreach ($pluginversions as $plugin) { foreach ($plugin_version as $plugin) {
$plugins[strtolower($plugin['name'])] = 1; $plugins[str_replace(' ', '', strtolower($plugin['name']))] = true;
} }
return $plugins; return $plugins;
} }
/* /**
* Technically, the NodeInfo spec defines 'active' as 'signed in at least once', * The NodeInfo page
* but GNU social doesn't keep track of when users last logged in, so let's return *
* the number of users that 'posted at least once', I guess. * @return void
* @author Stéphane Bérubé <chimo@chromic.org>
* @author Diogo Cordeiro <diogo@fc.up.pt>
*/ */
public function showNodeInfo(): void
public function showNodeInfo()
{ {
$openRegistrations = $this->getRegistrationsStatus(); $openRegistrations = $this->getRegistrationsStatus();
$userCount = $this->getUserCount(); $userCount = $this->getUserCount();
@ -51,43 +86,208 @@ class Nodeinfo_2_0Action extends ApiAction
$inboundServices = $this->getInboundServices(); $inboundServices = $this->getInboundServices();
$outboundServices = $this->getOutboundServices(); $outboundServices = $this->getOutboundServices();
$metadata = $this->getMetadata();
/* Required NodeInfo fields
"version",
"software",
"protocols",
"services",
"openRegistrations",
"usage",
"metadata"
*/
$json = json_encode([ $json = json_encode([
// The schema version, must be 2.0.
'version' => '2.0', 'version' => '2.0',
// [Mandatory] Metadata about server software in use.
'software' => [ 'software' => [
'name' => 'gnusocial', 'name' => 'gnusocial', // The canonical name of this server software.
'version' => GNUSOCIAL_VERSION 'version' => GNUSOCIAL_VERSION // The version of this server software.
], ],
// The protocols supported on this server.
// The spec requires an array containing at least 1 item but we can't ensure that.
'protocols' => $protocols, 'protocols' => $protocols,
// TODO: Have plugins register services // The third party sites this server can connect to via their application API.
'services' => [ 'services' => [
// The third party sites this server can retrieve messages from for combined display with regular traffic.
'inbound' => $inboundServices, 'inbound' => $inboundServices,
// The third party sites this server can publish messages to on the behalf of a user.
'outbound' => $outboundServices 'outbound' => $outboundServices
], ],
// Whether this server allows open self-registration.
'openRegistrations' => $openRegistrations, 'openRegistrations' => $openRegistrations,
// Usage statistics for this server.
'usage' => [ 'usage' => [
'users' => [ 'users' => [
// The total amount of on this server registered users.
'total' => $userCount, 'total' => $userCount,
// The amount of users that signed in at least once in the last 180 days.
'activeHalfyear' => $usersActiveHalfyear, 'activeHalfyear' => $usersActiveHalfyear,
// The amount of users that signed in at least once in the last 30 days.
'activeMonth' => $usersActiveMonth 'activeMonth' => $usersActiveMonth
], ],
// The amount of posts that were made by users that are registered on this server.
'localPosts' => $postCount, 'localPosts' => $postCount,
// The amount of comments that were made by users that are registered on this server.
'localComments' => $commentCount 'localComments' => $commentCount
], ],
'metadata' => new stdClass() // Free form key value pairs for software specific values. Clients should not rely on any specific key present.
'metadata' => $metadata
]); ]);
$this->initDocument('json'); header('Content-Type: application/json; profile=http://nodeinfo.diaspora.software/ns/schema/2.0#; charset=utf-8');
print $json; print $json;
$this->endDocument('json');
} }
public function getRegistrationsStatus() /**
* The protocols supported on this server.
* The spec requires an array containing at least 1 item but we can't ensure that
*
* These can only be one of:
* - activitypub,
* - buddycloud,
* - dfrn,
* - diaspora,
* - libertree,
* - ostatus,
* - pumpio,
* - tent,
* - xmpp,
* - zot
*
* @return array
* @author Diogo Cordeiro <diogo@fc.up.pt>
*/
public function getProtocols(): array
{
$protocols = [];
Event::handle('NodeInfoProtocols', [&$protocols]);
return $protocols;
}
/**
* The third party sites this server can retrieve messages from for combined display with regular traffic.
*
* These can only be one of:
* - atom1.0,
* - gnusocial,
* - imap,
* - pnut,
* - pop3,
* - pumpio,
* - rss2.0,
* - twitter
*
* @return array
* @author Diogo Cordeiro <diogo@fc.up.pt>
* @author Stéphane Bérubé <chimo@chromic.org>
*/
public function getInboundServices(): array
{
$inboundServices = [];
$ostatusEnabled = array_key_exists('ostatus', $this->plugins);
// We need those two to read feeds (despite WebSub).
if ($ostatusEnabled && array_key_exists('feedpoller', $this->plugins)) {
$inboundServices[] = 'atom1.0';
$inboundServices[] = 'rss2.0';
}
if (array_key_exists('twitterbridge', $this->plugins) && common_config('twitterimport', 'enabled')) {
$inboundServices[] = 'twitter';
}
if (array_key_exists('imap', $this->plugins)) {
$inboundServices[] = 'imap';
}
// We can receive messages from another GNU social instance if we have at least one of those enabled.
// And the same happens in the other instance
if ($ostatusEnabled || array_key_exists('activitypub', $this->plugins)) {
$inboundServices[] = 'gnusocial';
}
return $inboundServices;
}
/**
* The third party sites this server can publish messages to on the behalf of a user.
*
* These can only be one of:
* - atom1.0,
* - blogger,
* - buddycloud,
* - diaspora,
* - dreamwidth,
* - drupal,
* - facebook,
* - friendica,
* - gnusocial,
* - google,
* - insanejournal,
* - libertree,
* - linkedin,
* - livejournal,
* - mediagoblin,
* - myspace,
* - pinterest,
* - pnut,
* - posterous,
* - pumpio,
* - redmatrix,
* - rss2.0,
* - smtp,
* - tent,
* - tumblr,
* - twitter,
* - wordpress,
* - xmpp
*
* @return array
* @author Diogo Cordeiro <diogo@fc.up.pt>
* @author Stéphane Bérubé <chimo@chromic.org>
*/
public function getOutboundServices(): array
{
// Those two are always available
$outboundServices = ['atom1.0', 'rss2.0'];
if (array_key_exists('twitterbridge', $this->plugins)) {
$outboundServices[] = 'twitter';
}
// We can send messages to another GNU social instance if we have at least one of those enabled.
// And the same happens in the other instance
if (array_key_exists('ostatus', $this->plugins) ||
array_key_exists('activitypub', $this->plugins)) {
$outboundServices[] = 'gnusocial';
}
$xmppEnabled = (array_key_exists('xmpp', $this->plugins) && common_config('xmpp', 'enabled')) ? true : false;
if ($xmppEnabled) {
$outboundServices[] = 'xmpp';
}
return $outboundServices;
}
/**
* Whether this server allows open self-registration.
*
* @return bool
* @author Stéphane Bérubé <chimo@chromic.org>
*/
public function getRegistrationsStatus(): bool
{ {
$areRegistrationsClosed = (common_config('site', 'closed')) ? true : false; $areRegistrationsClosed = (common_config('site', 'closed')) ? true : false;
$isSiteInviteOnly = (common_config('site', 'inviteonly')) ? true : false; $isSiteInviteOnly = (common_config('site', 'inviteonly')) ? true : false;
@ -95,7 +295,13 @@ class Nodeinfo_2_0Action extends ApiAction
return !($areRegistrationsClosed || $isSiteInviteOnly); return !($areRegistrationsClosed || $isSiteInviteOnly);
} }
public function getUserCount() /**
* The total amount of on this server registered users.
*
* @return int
* @author Stéphane Bérubé <chimo@chromic.org>
*/
public function getUserCount(): int
{ {
$users = new Usage_stats(); $users = new Usage_stats();
$userCount = $users->getUserCount(); $userCount = $users->getUserCount();
@ -103,7 +309,46 @@ class Nodeinfo_2_0Action extends ApiAction
return $userCount; return $userCount;
} }
public function getPostCount() /**
* The amount of users that were active at least once in the last $days days.
*
* Technically, the NodeInfo spec defines 'active' as 'signed in at least once in the
* last {180, 30} days depending on request', but GNU social doesn't keep track of when
* users last logged in.
*
* Therefore, we use Favourites, Notices and Date of account creation to underestimate a
* value. Underestimate because a user that only logs in to see his feed is too an active
* user.
*
* @param int $days
* @return int
* @author Diogo Cordeiro <diogo@fc.up.pt>
*/
public function getActiveUsers(int $days): int
{
$query = "
SELECT COUNT(DISTINCT profile_id) as active_users_count
FROM (
SELECT profile_id FROM notice WHERE notice.created >= NOW() - INTERVAL {$days} DAY AND notice.is_local = 1
UNION ALL
SELECT user_id FROM fave INNER JOIN user ON fave.user_id = user.id WHERE fave.created >= NOW() - INTERVAL {$days} DAY
UNION ALL
SELECT id FROM user WHERE user.created >= NOW() - INTERVAL {$days} DAY
) as source";
$activeUsersCount = new DB_DataObject();
$activeUsersCount->query($query);
$activeUsersCount->fetch();
return $activeUsersCount->active_users_count;
}
/**
* The amount of posts that were made by users that are registered on this server.
*
* @return int
* @author Stéphane Bérubé <chimo@chromic.org>
*/
public function getPostCount(): int
{ {
$posts = new Usage_stats(); $posts = new Usage_stats();
$postCount = $posts->getPostCount(); $postCount = $posts->getPostCount();
@ -111,7 +356,13 @@ class Nodeinfo_2_0Action extends ApiAction
return $postCount; return $postCount;
} }
public function getCommentCount() /**
* The amount of comments that were made by users that are registered on this server.
*
* @return int
* @author Stéphane Bérubé <chimo@chromic.org>
*/
public function getCommentCount(): int
{ {
$comments = new Usage_stats(); $comments = new Usage_stats();
$commentCount = $comments->getCommentCount(); $commentCount = $comments->getCommentCount();
@ -119,61 +370,32 @@ class Nodeinfo_2_0Action extends ApiAction
return $commentCount; return $commentCount;
} }
public function getActiveUsers($days) /**
* Some additional information related to this GNU social instance
*
* @return array
* @author Diogo Cordeiro <diogo@fc.up.pt>
*/
public function getMetadata(): array
{ {
$notices = new Notice(); $metadata = [
$notices->joinAdd(array('profile_id', 'user:id')); 'nodeName' => common_config('site', 'name'),
$notices->whereAdd('notice.created >= NOW() - INTERVAL ' . $days . ' DAY'); 'software' => [
'homepage' => 'https://gnu.social/',
'repository' => 'https://notabug.org/diogo/gnu-social',
],
'uploadLimit' => common_get_preferred_php_upload_limit(),
'postFormats' => [
'text/plain',
'text/html'
],
'features' => []
];
$activeUsersCount = $notices->count('distinct profile_id'); if (array_key_exists('poll', $this->plugins)) {
$metadata['features'][] = 'polls';
return $activeUsersCount;
}
public function getProtocols()
{
$protocols = [];
Event::handle('NodeInfoProtocols', array(&$protocols));
return $protocols;
}
public function getInboundServices()
{
// FIXME: Are those always on?
$inboundServices = array('atom1.0', 'rss2.0');
if (array_key_exists('twitterbridge', $this->plugins) && common_config('twitterimport', 'enabled')) {
$inboundServices[] = 'twitter';
} }
if (array_key_exists('ostatus', $this->plugins)) { return $metadata;
$inboundServices[] = 'gnusocial';
}
return $inboundServices;
}
public function getOutboundServices()
{
$xmppEnabled = (array_key_exists('xmpp', $this->plugins) && common_config('xmpp', 'enabled')) ? true : false;
// FIXME: Are those always on?
$outboundServices = array('atom1.0', 'rss2.0');
if (array_key_exists('twitterbridge', $this->plugins)) {
$outboundServices[] = 'twitter';
}
if (array_key_exists('ostatus', $this->plugins)) {
$outboundServices[] = 'gnusocial';
}
if ($xmppEnabled) {
$outboundServices[] = 'xmpp';
}
return $outboundServices;
} }
} }

View File

@ -1,9 +1,36 @@
<?php <?php
// This file is part of GNU social - https://www.gnu.org/software/social
//
// GNU social 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.
//
// GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>.
if (!defined('GNUSOCIAL')) { /**
exit(1); * Provided in /.well-known/nodeinfo
} *
* @package NodeInfo
* @author Stéphane Bérubé <chimo@chromic.org>
* @copyright 2018-2019 Free Software Foundation, Inc http://www.fsf.org
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*/
defined('GNUSOCIAL') || die();
/**
* JRD document for NodeInfo
*
* @copyright 2018-2019 Free Software Foundation, Inc http://www.fsf.org
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*/
class NodeinfoJRDAction extends XrdAction class NodeinfoJRDAction extends XrdAction
{ {
const NODEINFO_2_0_REL = 'http://nodeinfo.diaspora.software/ns/schema/2.0'; const NODEINFO_2_0_REL = 'http://nodeinfo.diaspora.software/ns/schema/2.0';

View File

@ -1,72 +1,91 @@
<?php <?php
// This file is part of GNU social - https://www.gnu.org/software/social
//
// GNU social 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.
//
// GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>.
/** /**
* GNU social - a federating social network * Table for storing Nodeinfo statistics
* *
* LICENCE: This program is free software: you can redistribute it and/or modify * @package NodeInfo
* it under the terms of the GNU Affero General Public License as published by * @author Diogo Cordeiro <diogo@fc.up.pt>
* the Free Software Foundation, either version 3 of the License, or * @copyright 2018-2019 Free Software Foundation, Inc http://www.fsf.org
* (at your option) any later version. * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*
* 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/>.
*/ */
if (!defined('GNUSOCIAL')) { defined('GNUSOCIAL') || die();
exit(1);
}
/** /**
* Table Definition for Usage_stats * Table Definition for usage_stats and some getters
*
* @copyright 2018-2019 Free Software Foundation, Inc http://www.fsf.org
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*/ */
class Usage_stats extends Managed_DataObject class Usage_stats extends Managed_DataObject
{ {
###START_AUTOCODE
/* the code below is auto generated do not remove the above tag */
public $__table = 'usage_stats'; // table name public $__table = 'usage_stats'; // table name
public $type; // varchar(191) unique_key not 255 because utf8mb4 takes more space public $type; // varchar(191) unique_key not 255 because utf8mb4 takes more space
public $count; // int(4) public $count; // int(4)
public $modified; // timestamp() not_null default_CURRENT_TIMESTAMP public $modified; // datetime() not_null default_CURRENT_TIMESTAMP
/* the code above is auto generated do not remove the tag below */ /**
###END_AUTOCODE * Table Definition for usage_stats
*
public static function schemaDef() * @return array
*/
public static function schemaDef(): array
{ {
return [ return [
'description' => 'node stats', 'description' => 'node stats',
'fields' => [ 'fields' => [
'type' => ['type' => 'varchar', 'length' => 191, 'description' => 'Type of countable entity'], 'type' => ['type' => 'varchar', 'not null' => true, 'length' => 191, 'description' => 'Type of countable entity'],
'count' => ['type' => 'int', 'size' => 'int', 'default' => 0, 'description' => 'Number of entities of this type'], 'count' => ['type' => 'int', 'size' => 'int', 'default' => 0, 'description' => 'Number of entities of this type'],
'modified' => ['type' => 'timestamp', 'not null' => true, 'description' => 'date this record was modified'], 'modified' => ['type' => 'datetime', 'not null' => true, 'default' => 'CURRENT_TIMESTAMP', 'description' => 'date this record was modified'],
], ],
'primary key' => ['type'], 'primary key' => ['type'],
'unique keys' => [
'usage_stats_key' => ['type'],
],
'indexes' => [ 'indexes' => [
'user_stats_idx' => ['type'], 'user_stats_idx' => ['type'],
], ],
]; ];
} }
public function getUserCount() /**
* Total number of users
*
* @return int
*/
public function getUserCount(): int
{ {
return intval(Usage_stats::getKV('type', 'users')->count); return Usage_stats::getKV('type', 'users')->count;
} }
public function getPostCount() /**
* Total number of dents
*
* @return int
*/
public function getPostCount(): int
{ {
return intval(Usage_stats::getKV('type', 'posts')->count); return Usage_stats::getKV('type', 'posts')->count;
} }
public function getCommentCount() /**
* Total number of replies
*
* @return int
*/
public function getCommentCount(): int
{ {
return intval(Usage_stats::getKV('type', 'comments')->count); return Usage_stats::getKV('type', 'comments')->count;
} }
} }

View File

@ -1,32 +1,32 @@
#!/usr/bin/env php #!/usr/bin/env php
<?php <?php
// This file is part of GNU social - https://www.gnu.org/software/social
//
// GNU social 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.
//
// GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>.
/** /**
* GNU social - a federating social network * Fix Nodeinfo statistics
* *
* LICENCE: This program is free software: you can redistribute it and/or modify * @package NodeInfo
* it under the terms of the GNU Affero General Public License as published by * @author Diogo Cordeiro <diogo@fc.up.pt>
* the Free Software Foundation, either version 3 of the License, or * @copyright 2019 Free Software Foundation, Inc http://www.fsf.org
* (at your option) any later version. * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
*
* 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/>.
*
* @category Plugin
* @package GNUsocial
* @copyright 2018 Free Software Foundation http://fsf.org
* @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
* @link https://www.gnu.org/software/social/
*/ */
define('INSTALLDIR', realpath(__DIR__ . '/../../..')); define('INSTALLDIR', dirname(dirname(dirname(__DIR__))));
if (!defined('NODEINFO_UPGRADE')) { if (!defined('NODEINFO_UPGRADE')) {
$longoptions = ['type=']; $longoptions = ['type='];
$helptext = <<<END_OF_HELP $helptext = <<<END_OF_HELP
@ -53,7 +53,6 @@ END_OF_HELP;
if ($verbose) { if ($verbose) {
echo "Started.\n\n"; echo "Started.\n\n";
} }
} else { } else {
echo "Nodeinfo will now fix stats\n"; echo "Nodeinfo will now fix stats\n";
$type_to_fix = 'all'; $type_to_fix = 'all';
@ -95,7 +94,13 @@ if ($verbose) {
* Counting functions * Counting functions
*/ */
function getUserCount() /**
* Total number of users
*
* @return int
* @author Stéphane Bérubé <chimo@chromic.org>
*/
function getUserCount(): int
{ {
$users = new User(); $users = new User();
$userCount = $users->count(); $userCount = $users->count();
@ -103,6 +108,12 @@ function getUserCount()
return $userCount; return $userCount;
} }
/**
* Total number of dents
*
* @return int
* @author Stéphane Bérubé <chimo@chromic.org>
*/
function getPostCount() function getPostCount()
{ {
$notices = new Notice(); $notices = new Notice();
@ -113,6 +124,12 @@ function getPostCount()
return $noticeCount; return $noticeCount;
} }
/**
* Total number of replies
*
* @return int
* @author Stéphane Bérubé <chimo@chromic.org>
*/
function getCommentCount() function getCommentCount()
{ {
$notices = new Notice(); $notices = new Notice();