2019-04-20 14:15:45 +02:00
< ? php
/**
2019-04-20 20:38:32 +02:00
* Name : blockbot
2019-04-20 14:15:45 +02:00
* Description : Blocking bots based on detecting bots / crawlers / spiders via the user agent and http_from header .
2019-07-28 07:49:30 +00:00
* Version : 0.2
2019-04-20 14:15:45 +02:00
* Author : Philipp Holzer < admin @ philipp . info >
2019-07-28 07:49:30 +00:00
* Author : Michael Vogel < https :// pirati . ca / profile / heluecht >
2019-04-20 14:15:45 +02:00
*
*/
use Friendica\Core\Hook ;
2020-01-18 22:07:06 +01:00
use Friendica\DI ;
2019-04-20 14:15:45 +02:00
use Jaybizzle\CrawlerDetect\CrawlerDetect ;
2019-04-27 13:51:44 +02:00
use Friendica\Core\Logger ;
2019-07-28 07:49:30 +00:00
use Friendica\Core\Renderer ;
2021-11-04 20:32:16 +00:00
use Friendica\Network\HTTPException\ForbiddenException ;
2019-04-20 14:15:45 +02:00
2019-04-21 12:35:33 +02:00
require_once __DIR__ . DIRECTORY_SEPARATOR . 'vendor' . DIRECTORY_SEPARATOR . 'autoload.php' ;
2022-06-23 07:16:22 +02:00
function blockbot_install ()
{
2019-04-22 10:49:40 +02:00
Hook :: register ( 'init_1' , __FILE__ , 'blockbot_init_1' );
2019-04-20 14:15:45 +02:00
}
2023-01-13 21:16:09 -05:00
function blockbot_addon_admin ( string & $o )
2022-06-23 07:16:22 +02:00
{
$t = Renderer :: getMarkupTemplate ( 'admin.tpl' , 'addon/blockbot/' );
2019-07-28 07:49:30 +00:00
$o = Renderer :: replaceMacros ( $t , [
2024-03-04 05:37:04 +00:00
'$submit' => DI :: l10n () -> t ( 'Save Settings' ),
'$good_crawlers' => [ 'good_crawlers' , DI :: l10n () -> t ( 'Allow "good" crawlers' ), DI :: config () -> get ( 'blockbot' , 'good_crawlers' ), DI :: l10n () -> t ( " Don't block fediverse crawlers, relay servers and other bots with good purposes. " )],
'$socialmedia_agents' => [ 'socialmedia_agents' , DI :: l10n () -> t ( 'Allow preview agents' ), DI :: config () -> get ( 'blockbot' , 'socialmedia_agents' ), DI :: l10n () -> t ( " Don't block agents from social media systems that want to generate preview data for links that had been set by their users. " )],
'$block_gab' => [ 'block_gab' , DI :: l10n () -> t ( 'Block GabSocial' ), DI :: config () -> get ( 'blockbot' , 'block_gab' ), DI :: l10n () -> t ( 'Block the software GabSocial. This will block every access for that software. You can block dedicated gab instances in the blocklist settings in the admin section.' )],
'$training' => [ 'training' , DI :: l10n () -> t ( 'Training mode' ), DI :: config () -> get ( 'blockbot' , 'training' ), DI :: l10n () -> t ( " Activates the training mode. This is only meant for developing purposes. Don't activate this on a production machine. This can cut communication with some systems. " )],
2019-07-28 07:49:30 +00:00
]);
}
2023-01-13 21:16:09 -05:00
function blockbot_addon_admin_post ()
2022-06-23 07:16:22 +02:00
{
2020-01-19 21:21:52 +01:00
DI :: config () -> set ( 'blockbot' , 'good_crawlers' , $_POST [ 'good_crawlers' ] ? ? false );
2024-03-04 05:37:04 +00:00
DI :: config () -> set ( 'blockbot' , 'socialmedia_agents' , $_POST [ 'socialmedia_agents' ] ? ? false );
2020-01-19 21:21:52 +01:00
DI :: config () -> set ( 'blockbot' , 'block_gab' , $_POST [ 'block_gab' ] ? ? false );
DI :: config () -> set ( 'blockbot' , 'training' , $_POST [ 'training' ] ? ? false );
2019-07-28 07:49:30 +00:00
}
2023-01-13 21:16:09 -05:00
function blockbot_init_1 ()
2022-06-23 07:16:22 +02:00
{
2019-05-03 12:25:13 +02:00
if ( empty ( $_SERVER [ 'HTTP_USER_AGENT' ])) {
return ;
}
2019-04-27 15:34:51 +02:00
$logdata = [ 'agent' => $_SERVER [ 'HTTP_USER_AGENT' ], 'uri' => $_SERVER [ 'REQUEST_URI' ]];
2019-07-28 08:13:53 +00:00
// List of "good" crawlers
2023-03-05 14:01:32 +00:00
$good_agents = [
'fediverse.space crawler' , 'fediverse.network crawler' , 'Active_Pods_CheckBot_3.0' ,
2020-09-06 07:30:59 +00:00
'Social-Relay/' , 'Test Certificate Info' , 'Uptimebot/' , 'GNUSocialBot' , 'UptimeRobot/' ,
2024-03-04 05:37:04 +00:00
'PTST/' , 'Zabbix' , 'Poduptime/' , 'FediFetcher' , 'lemmy-stats-crawler' ,
'FedditLemmyverseCrawler/' , 'kbinBot/' , 'lemmy-explorer-crawler/' ,
2023-03-05 14:01:32 +00:00
];
2019-07-28 08:13:53 +00:00
2024-03-04 05:37:04 +00:00
// List of agents from social media systems that fetch preview data via opem graph or twitter cards
$socialmedia_agents = [ 'Twitterbot/' , 'facebookexternalhit/' , 'SkypeUriPreview Preview/' ,
'TelegramBot' , 'WhatsApp/' , 'github-camo' , 'Bluesky Cardyb/' , 'XING-contenttabreceiver/' ,
'LinkedInBot/' , 'Instagram ' , 'Synapse (bot; ' , 'Discordbot/' ];
// List of known unwanted crawlers.
2023-03-05 14:01:32 +00:00
$agents = [
'SemrushBot' , 's~feedly-nikon3' , 'Qwantify/Bleriot/' , 'ltx71' , 'Sogou web spider/' ,
2024-03-04 05:37:04 +00:00
'Diffbot/' , 'YisouSpider' , 'evc-batch/' , 'LivelapBot/' , 'TrendsmapResolver/' ,
2019-05-29 18:51:07 +00:00
'PaperLiBot/' , 'Nuzzel' , 'um-LN/' , 'Google Favicon' , 'Datanyze' , 'BLEXBot/' , '360Spider' ,
'adscanner/' , 'HeadlessChrome' , 'wpif' , 'startmebot/' , 'Googlebot/' , 'Applebot/' ,
2024-03-04 05:37:04 +00:00
'GoogleImageProxy' , 'bingbot/' , 'heritrix/' , 'ldspider' ,
2021-05-24 06:21:07 +00:00
'AwarioRssBot/' , 'TweetmemeBot/' , 'dcrawl/' , 'PhantomJS/' , 'Googlebot-Image/' ,
2019-06-01 04:51:01 +00:00
'CrowdTanglebot/' , 'Mediapartners-Google' , 'Baiduspider/' , 'datagnionbot' ,
2019-06-06 20:31:16 +00:00
'MegaIndex.ru/' , 'SMUrlExpander' , 'Hatena-Favicon/' , 'Wappalyzer' , 'FlipboardProxy/' ,
'NetcraftSurveyAgent/' , 'Dataprovider.com' , 'SMTBot/' , 'Nimbostratus-Bot/' ,
2019-06-10 14:33:42 +00:00
'DuckDuckGo-Favicons-Bot/' , 'IndieWebCards/' , 'proximic' , 'netEstate NE Crawler' ,
2024-03-04 05:37:04 +00:00
'AhrefsBot/' , 'YandexBot/' , 'Exabot/' , 'Mediumbot-MetaTagFetcher/' ,
'SurdotlyBot/' , 'BingPreview/' , 'SabsimBot/' , 'CCBot/' , 'WbSrch/' ,
2019-06-20 05:31:53 +00:00
'DuckDuckBot-Https/' , 'HTTP Banner Detection' , 'YandexImages/' , 'archive.org_bot' ,
'ArchiveTeam ArchiveBot/' , 'yacybot' , 'https://developers.google.com/+/web/snippet/' ,
2024-03-04 05:37:04 +00:00
'Scrapy/' , 'MJ12bot/' , 'DotBot/' , 'Pinterestbot/' , 'Jooblebot/' ,
2019-07-10 02:16:57 +00:00
'Cliqzbot/' , 'YaK/' , 'Mediatoolkitbot' , 'Snacktory' , 'FunWebProducts' , 'oBot/' ,
2019-07-29 15:48:51 +00:00
'7Siters/' , 'KOCMOHABT' , 'Google-SearchByImage' , 'FemtosearchBot/' ,
2020-07-05 16:36:17 +02:00
'HubSpot Crawler' , 'DomainStatsBot/' , 'Re-re Studio' , 'AwarioSmartBot/' ,
2020-07-07 18:51:42 +00:00
'SummalyBot/' , 'DNSResearchBot/' , 'PetalBot;' , 'Nmap Scripting Engine;' ,
2020-07-09 15:08:41 +00:00
'Google-Apps-Script; beanserver;' , 'woorankreview/' , 'Seekport Crawler;' , 'AHC/' ,
2024-03-04 05:37:04 +00:00
'Semanticbot/' , 'Embed PHP library' , 'XoviOnpageCrawler;' , 'Pinterest/' ,
2021-08-11 07:50:55 +00:00
'GetHPinfo.com-Bot/' , 'BoardReader Favicon Fetcher' , 'Google-Adwords-Instant' , 'newspaper/' ,
2024-03-04 05:37:04 +00:00
'YurichevBot/' , 'Crawling at Home Project' , 'InfoTigerBot/' ,
'AdIdxBot/' , 'MicrosoftPreview/' , 'masscan/'
2023-03-05 14:01:32 +00:00
];
2019-07-28 07:49:30 +00:00
2020-01-19 21:21:12 +01:00
if ( ! DI :: config () -> get ( 'blockbot' , 'good_crawlers' )) {
2019-07-28 08:13:53 +00:00
$agents = array_merge ( $agents , $good_agents );
2020-01-18 23:28:14 +01:00
} else {
foreach ( $good_agents as $good_agent ) {
2020-01-18 13:47:20 +01:00
if ( stristr ( $_SERVER [ 'HTTP_USER_AGENT' ], $good_agent )) {
return ;
}
}
}
2019-07-28 08:13:53 +00:00
2024-03-04 05:37:04 +00:00
if ( ! DI :: config () -> get ( 'blockbot' , 'socialmedia_agents' )) {
$agents = array_merge ( $agents , $socialmedia_agents );
} else {
foreach ( $socialmedia_agents as $socialmedia_agent ) {
if ( stristr ( $_SERVER [ 'HTTP_USER_AGENT' ], $socialmedia_agent )) {
return ;
}
}
}
2020-01-19 21:21:12 +01:00
if ( DI :: config () -> get ( 'blockbot' , 'block_gab' )) {
2019-07-28 07:49:30 +00:00
$agents [] = 'GabSocial/' ;
}
2019-05-29 18:51:07 +00:00
foreach ( $agents as $agent ) {
if ( stristr ( $_SERVER [ 'HTTP_USER_AGENT' ], $agent )) {
2024-03-04 05:37:04 +00:00
throw new ForbiddenException ( 'Bots are not allowed. If you consider this a mistake, create an issue at https://github.com/friendica/friendica' );
2019-05-29 18:51:07 +00:00
}
}
2019-05-30 06:45:20 +02:00
// This switch here is only meant for developers who want to add more bots to the list above, it is not safe for production.
2020-01-19 21:21:12 +01:00
if ( ! DI :: config () -> get ( 'blockbot' , 'training' )) {
2019-05-29 18:51:07 +00:00
return ;
}
$crawlerDetect = new CrawlerDetect ();
2019-04-27 15:34:51 +02:00
if ( ! $crawlerDetect -> isCrawler ()) {
logger :: debug ( 'Good user agent detected' , $logdata );
return ;
}
2019-04-29 22:21:42 +02:00
// List of false positives' strings of known "good" agents.
2023-03-05 14:01:32 +00:00
$agents = [
'curl' , 'zgrab' , 'Go-http-client' , 'curb' , 'github.com' , 'reqwest' , 'Feedly/' ,
2019-04-29 22:21:42 +02:00
'Python-urllib/' , 'Liferea/' , 'aiohttp/' , 'WordPress.com Reader' , 'hackney/' ,
'Faraday v' , 'okhttp' , 'UniversalFeedParser' , 'PixelFedBot' , 'python-requests' ,
2019-05-30 10:32:01 +00:00
'WordPress/' , 'http.rb/' , 'Apache-HttpClient/' , 'WordPress.com;' , 'Pleroma' ,
2020-07-12 10:41:29 +02:00
'Dispatch/' , 'Ruby' , 'Java/' , 'libwww-perl/' , 'Mastodon/' , 'FeedlyApp/' ,
2020-07-08 19:24:00 +00:00
'lua-resty-http/' , 'Tiny Tiny RSS/' , 'Wget/' , 'PostmanRuntime/' ,
2021-08-11 07:50:55 +00:00
'W3C_Validator/' , 'NetNewsWire' , 'FeedValidator/' , 'theoldreader.com' , 'axios/' ,
2021-12-08 20:00:45 +00:00
'Paw/' , 'PeerTube/' , 'fedi.inex.dev' , 'FediDB/' , 'index.community crawler' ,
2024-03-04 05:37:04 +00:00
'Slackbot-LinkExpanding' , 'Firefish/' , 'Takahe/' , 'Akkoma ' , 'Misskey/'
2023-03-05 14:01:32 +00:00
];
2019-07-28 08:13:53 +00:00
2020-01-19 21:21:12 +01:00
if ( DI :: config () -> get ( 'blockbot' , 'good_crawlers' )) {
2019-07-28 08:13:53 +00:00
$agents = array_merge ( $agents , $good_agents );
}
2019-04-27 15:34:51 +02:00
foreach ( $agents as $agent ) {
if ( stristr ( $_SERVER [ 'HTTP_USER_AGENT' ], $agent )) {
2023-03-05 14:01:32 +00:00
logger :: info ( 'False positive' , $logdata );
2019-04-27 15:34:51 +02:00
return ;
}
}
2023-03-05 14:01:32 +00:00
logger :: notice ( 'Blocked bot' , $logdata );
2024-03-04 05:37:04 +00:00
throw new ForbiddenException ( 'Bots are not allowed. If you consider this a mistake, create an issue at https://github.com/friendica/friendica' );
2019-04-20 14:15:45 +02:00
}