]> git.mxchange.org Git - friendica.git/blob - src/App/Router.php
Move mod/directory to src/Module/Directory
[friendica.git] / src / App / Router.php
1 <?php
2
3 namespace Friendica\App;
4
5
6 use FastRoute\DataGenerator\GroupCountBased;
7 use FastRoute\Dispatcher;
8 use FastRoute\RouteCollector;
9 use FastRoute\RouteParser\Std;
10 use Friendica\Module;
11
12 /**
13  * Wrapper for FastRoute\Router
14  *
15  * This wrapper only makes use of a subset of the router features, mainly parses a route rule to return the relevant
16  * module class.
17  *
18  * Actual routes are defined in App->collectRoutes.
19  *
20  * @package Friendica\App
21  */
22 class Router
23 {
24         /** @var RouteCollector */
25         protected $routeCollector;
26
27         /**
28          * Static declaration of Friendica routes.
29          *
30          * Supports:
31          * - Route groups
32          * - Variable parts
33          * Disregards:
34          * - HTTP method other than GET
35          * - Named parameters
36          *
37          * Handler must be the name of a class extending Friendica\BaseModule.
38          *
39          * @brief Static declaration of Friendica routes.
40          */
41         public function collectRoutes()
42         {
43                 $this->routeCollector->addRoute(['GET', 'POST'], '/itemsource[/{guid}]', Module\Itemsource::class);
44                 $this->routeCollector->addRoute(['GET'],         '/amcd',                Module\AccountManagementControlDocument::class);
45                 $this->routeCollector->addGroup('/.well-known', function (RouteCollector $collector) {
46                         $collector->addRoute(['GET'], '/host-meta'       , Module\WellKnown\HostMeta::class);
47                         $collector->addRoute(['GET'], '/nodeinfo[/1.0]'  , Module\NodeInfo::class);
48                         $collector->addRoute(['GET'], '/webfinger'       , Module\Xrd::class);
49                         $collector->addRoute(['GET'], '/x-social-relay'  , Module\WellKnown\XSocialRelay::class);
50                 });
51                 $this->routeCollector->addRoute(['GET'],         '/acctlink',            Module\Acctlink::class);
52                 $this->routeCollector->addRoute(['GET'],         '/apps',                Module\Apps::class);
53                 $this->routeCollector->addRoute(['GET'],         '/attach/{item:\d+}',   Module\Attach::class);
54                 $this->routeCollector->addRoute(['GET'],         '/babel',               Module\Babel::class);
55                 $this->routeCollector->addGroup('/contact', function (RouteCollector $collector) {
56                         $collector->addRoute(['GET'], '[/]',                                 Module\Contact::class);
57                         $collector->addRoute(['GET'], '/{id:\d+}[/posts|conversations]',     Module\Contact::class);
58                 });
59                 $this->routeCollector->addRoute(['GET'],         '/credits',             Module\Credits::class);
60                 $this->routeCollector->addGroup('/feed', function (RouteCollector $collector) {
61                         $collector->addRoute(['GET'], '/{nickname}',                         Module\Feed::class);
62                         $collector->addRoute(['GET'], '/{nickname}/posts',                   Module\Feed::class);
63                         $collector->addRoute(['GET'], '/{nickname}/comments',                Module\Feed::class);
64                         $collector->addRoute(['GET'], '/{nickname}/replies',                 Module\Feed::class);
65                         $collector->addRoute(['GET'], '/{nickname}/activity',                Module\Feed::class);
66                 });
67                 $this->routeCollector->addRoute(['GET'],         '/directory',           Module\Directory::class);
68                 $this->routeCollector->addRoute(['GET'],         '/feedtest',            Module\Feedtest::class);
69                 $this->routeCollector->addRoute(['GET'],         '/filer[/{id:\d+}]',    Module\Filer::class);
70                 $this->routeCollector->addRoute(['GET'],         '/followers/{owner}',   Module\Followers::class);
71                 $this->routeCollector->addRoute(['GET'],         '/following/{owner}',   Module\Following::class);
72                 $this->routeCollector->addGroup('/group', function (RouteCollector $collector) {
73                         $collector->addRoute(['GET', 'POST'], '[/]',                         Module\Group::class);
74                         $collector->addRoute(['GET', 'POST'], '/{group:\d+}',                Module\Group::class);
75                         $collector->addRoute(['GET', 'POST'], '/none',                       Module\Group::class);
76                         $collector->addRoute(['GET', 'POST'], '/new',                        Module\Group::class);
77                         $collector->addRoute(['GET', 'POST'], '/drop/{group:\d+}',           Module\Group::class);
78                         $collector->addRoute(['GET', 'POST'], '/{group:\d+}/{contact:\d+}',  Module\Group::class);
79
80                         $collector->addRoute(['POST'], '/{group:\d+}/add/{contact:\d+}',     Module\Group::class);
81                         $collector->addRoute(['POST'], '/{group:\d+}/remove/{contact:\d+}',  Module\Group::class);
82                 });
83                 $this->routeCollector->addRoute(['GET'],         '/hashtag',             Module\Hashtag::class);
84                 $this->routeCollector->addRoute(['GET'],         '/inbox[/{nickname}]',  Module\Inbox::class);
85                 $this->routeCollector->addGroup('/install', function (RouteCollector $collector) {
86                         $collector->addRoute(['GET', 'POST'], '[/]',                         Module\Install::class);
87                         $collector->addRoute(['GET'],         '/testrewrite',                Module\Install::class);
88                 });
89                 $this->routeCollector->addRoute(['GET', 'POST'], '/localtime',           Module\Localtime::class);
90                 $this->routeCollector->addRoute(['GET', 'POST'], '/login',               Module\Login::class);
91                 $this->routeCollector->addRoute(['GET'],         '/magic',               Module\Magic::class);
92                 $this->routeCollector->addRoute(['GET'],         '/manifest',            Module\Manifest::class);
93                 $this->routeCollector->addRoute(['GET'],         '/nodeinfo/1.0',        Module\NodeInfo::class);
94                 $this->routeCollector->addRoute(['GET'],         '/objects/{guid}',      Module\Objects::class);
95                 $this->routeCollector->addGroup('/oembed', function (RouteCollector $collector) {
96                         $collector->addRoute(['GET'], '/[b2h|h2b]',                          Module\Oembed::class);
97                         $collector->addRoute(['GET'], '/{hash}',                             Module\Oembed::class);
98                 });
99                 $this->routeCollector->addRoute(['GET'],         '/outbox/{owner}',      Module\Outbox::class);
100                 $this->routeCollector->addRoute(['GET'],         '/owa',                 Module\Owa::class);
101                 $this->routeCollector->addGroup('/photo', function (RouteCollector $collector) {
102                         $collector->addRoute(['GET'], '/{name}',                             Module\Photo::class);
103                         $collector->addRoute(['GET'], '/{type}/{name}',                      Module\Photo::class);
104                         $collector->addRoute(['GET'], '/{type}/{customize}/{name}',          Module\Photo::class);
105                 });
106                 $this->routeCollector->addGroup('/profile', function (RouteCollector $collector) {
107                         $collector->addRoute(['GET'], '/{nickname}',                         Module\Profile::class);
108                         $collector->addRoute(['GET'], '/{profile:\d+}/view',                 Module\Profile::class);
109                 });
110                 $this->routeCollector->addGroup('/proxy', function (RouteCollector $collector) {
111                         $collector->addRoute(['GET'], '[/]',                                 Module\Proxy::class);
112                         $collector->addRoute(['GET'], '/{url}',                              Module\Proxy::class);
113                         $collector->addRoute(['GET'], '/sub1/{url}',                         Module\Proxy::class);
114                         $collector->addRoute(['GET'], '/sub1/sub2/{url}',                    Module\Proxy::class);
115                 });
116                 $this->routeCollector->addRoute(['GET', 'POST'], '/register',            Module\Register::class);
117                 $this->routeCollector->addRoute(['GET'],         '/statistics.json',     Module\Statistics::class);
118                 $this->routeCollector->addRoute(['GET'],         '/tos',                 Module\Tos::class);
119                 $this->routeCollector->addRoute(['GET'],         '/webfinger',           Module\WebFinger::class);
120                 $this->routeCollector->addRoute(['GET'],         '/xrd',                 Module\Xrd::class);
121
122                 $this->routeCollector->addGroup('/admin', function (RouteCollector $collector) {
123                         $collector->addRoute(['GET']        , '[/]'                     , Module\Admin\Summary::class);
124
125                         $collector->addRoute(['GET', 'POST'], '/addons'                 , Module\Admin\Addons\Index::class);
126                         $collector->addRoute(['GET', 'POST'], '/addons/{addon}'         , Module\Admin\Addons\Details::class);
127
128                         $collector->addRoute(['GET', 'POST'], '/blocklist/contact'      , Module\Admin\Blocklist\Contact::class);
129                         $collector->addRoute(['GET', 'POST'], '/blocklist/server'       , Module\Admin\Blocklist\Server::class);
130
131                         $collector->addRoute(['GET']        , '/dbsync[/check]'         , Module\Admin\DBSync::class);
132                         $collector->addRoute(['GET']        , '/dbsync/{update:\d+}'    , Module\Admin\DBSync::class);
133                         $collector->addRoute(['GET']        , '/dbsync/mark/{update:\d+}', Module\Admin\DBSync::class);
134
135                         $collector->addRoute(['GET', 'POST'], '/features'               , Module\Admin\Features::class);
136                         $collector->addRoute(['GET']        , '/federation'             , Module\Admin\Federation::class);
137
138                         $collector->addRoute(['GET', 'POST'], '/item/delete'            , Module\Admin\Item\Delete::class);
139                         $collector->addRoute(['GET', 'POST'], '/item/source[/{guid}]'   , Module\Admin\Item\Source::class);
140
141                         $collector->addRoute(['GET']        , '/logs/view'              , Module\Admin\Logs\View::class);
142                         $collector->addRoute(['GET', 'POST'], '/logs'                   , Module\Admin\Logs\Settings::class);
143
144                         $collector->addRoute(['GET']        , '/phpinfo'                , Module\Admin\PhpInfo::class);
145
146                         $collector->addRoute(['GET']        , '/queue[/deferred]'       , Module\Admin\Queue::class);
147
148                         $collector->addRoute(['GET', 'POST'], '/site'                   , Module\Admin\Site::class);
149
150                         $collector->addRoute(['GET', 'POST'], '/themes'                 , Module\Admin\Themes\Index::class);
151                         $collector->addRoute(['GET', 'POST'], '/themes/{theme}'         , Module\Admin\Themes\Details::class);
152                         $collector->addRoute(['GET', 'POST'], '/themes/{theme}/embed'   , Module\Admin\Themes\Embed::class);
153
154                         $collector->addRoute(['GET', 'POST'], '/tos'                    , Module\Admin\Tos::class);
155
156                         $collector->addRoute(['GET', 'POST'], '/users[/{action}/{uid}]' , Module\Admin\Users::class);
157                 });
158         }
159
160         public function __construct(RouteCollector $routeCollector = null)
161         {
162                 if (!$routeCollector) {
163                         $routeCollector = new RouteCollector(new Std(), new GroupCountBased());
164                 }
165
166                 $this->routeCollector = $routeCollector;
167         }
168
169         public function getRouteCollector()
170         {
171                 return $this->routeCollector;
172         }
173
174         /**
175          * Returns the relevant module class name for the given page URI or NULL if no route rule matched.
176          *
177          * @param string $cmd The path component of the request URL without the query string
178          * @return string|null A Friendica\BaseModule-extending class name if a route rule matched
179          */
180         public function getModuleClass($cmd)
181         {
182                 $cmd = '/' . ltrim($cmd, '/');
183
184                 $dispatcher = new \FastRoute\Dispatcher\GroupCountBased($this->routeCollector->getData());
185
186                 $moduleClass = null;
187
188                 // @TODO: Enable method-specific modules
189                 $httpMethod = 'GET';
190                 $routeInfo = $dispatcher->dispatch($httpMethod, $cmd);
191                 if ($routeInfo[0] === Dispatcher::FOUND) {
192                         $moduleClass = $routeInfo[1];
193                 }
194
195                 return $moduleClass;
196         }
197 }