]> git.mxchange.org Git - quix0rs-gnu-social.git/blob - lib/router.php
fix local file include vulnerability in doc.php
[quix0rs-gnu-social.git] / lib / router.php
1 <?php
2 /**
3  * StatusNet, the distributed open-source microblogging tool
4  *
5  * URL routing utilities
6  *
7  * PHP version 5
8  *
9  * LICENCE: This program is free software: you can redistribute it and/or modify
10  * it under the terms of the GNU Affero General Public License as published by
11  * the Free Software Foundation, either version 3 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU Affero General Public License for more details.
18  *
19  * You should have received a copy of the GNU Affero General Public License
20  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
21  *
22  * @category  URL
23  * @package   StatusNet
24  * @author    Evan Prodromou <evan@status.net>
25  * @copyright 2009 StatusNet, Inc.
26  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
27  * @link      http://status.net/
28  */
29
30 if (!defined('STATUSNET') && !defined('LACONICA')) {
31     exit(1);
32 }
33
34 require_once 'Net/URL/Mapper.php';
35
36 /**
37  * URL Router
38  *
39  * Cheap wrapper around Net_URL_Mapper
40  *
41  * @category URL
42  * @package  StatusNet
43  * @author   Evan Prodromou <evan@status.net>
44  * @license  http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
45  * @link     http://status.net/
46  */
47
48 class Router
49 {
50     var $m = null;
51     static $inst = null;
52     static $bare = array('requesttoken', 'accesstoken', 'userauthorization',
53                          'postnotice', 'updateprofile', 'finishremotesubscribe');
54
55     static function get()
56     {
57         if (!Router::$inst) {
58             Router::$inst = new Router();
59         }
60         return Router::$inst;
61     }
62
63     function __construct()
64     {
65         if (!$this->m) {
66             $this->m = $this->initialize();
67         }
68     }
69
70     function initialize()
71     {
72         $m = Net_URL_Mapper::getInstance();
73
74         if (Event::handle('StartInitializeRouter', array(&$m))) {
75
76             $m->connect('robots.txt', array('action' => 'robotstxt'));
77
78             $m->connect('opensearch/people', array('action' => 'opensearch',
79                                                    'type' => 'people'));
80             $m->connect('opensearch/notice', array('action' => 'opensearch',
81                                                    'type' => 'notice'));
82
83             // docs
84
85             $m->connect('doc/:title', array('action' => 'doc'));
86
87             $m->connect('main/otp/:user_id/:token',
88                         array('action' => 'otp'),
89                         array('user_id' => '[0-9]+',
90                               'token' => '.+'));
91
92             // main stuff is repetitive
93
94             $main = array('login', 'logout', 'register', 'subscribe',
95                           'unsubscribe', 'confirmaddress', 'recoverpassword',
96                           'invite', 'favor', 'disfavor', 'sup',
97                           'block', 'unblock', 'subedit',
98                           'groupblock', 'groupunblock',
99                           'sandbox', 'unsandbox',
100                           'silence', 'unsilence',
101                           'repeat',
102                           'deleteuser',
103                           'geocode',
104                           'version',
105                           );
106
107             foreach ($main as $a) {
108                 $m->connect('main/'.$a, array('action' => $a));
109             }
110
111             $m->connect('main/sup/:seconds', array('action' => 'sup'),
112                         array('seconds' => '[0-9]+'));
113
114             $m->connect('main/tagother/:id', array('action' => 'tagother'));
115
116             $m->connect('main/oembed',
117                         array('action' => 'oembed'));
118
119             $m->connect('main/xrds',
120                         array('action' => 'publicxrds'));
121
122             // these take a code
123
124             foreach (array('register', 'confirmaddress', 'recoverpassword') as $c) {
125                 $m->connect('main/'.$c.'/:code', array('action' => $c));
126             }
127
128             // exceptional
129
130             $m->connect('main/remote', array('action' => 'remotesubscribe'));
131             $m->connect('main/remote?nickname=:nickname', array('action' => 'remotesubscribe'), array('nickname' => '[A-Za-z0-9_-]+'));
132
133             foreach (Router::$bare as $action) {
134                 $m->connect('index.php?action=' . $action, array('action' => $action));
135             }
136
137             // settings
138
139             foreach (array('profile', 'avatar', 'password', 'im', 'oauthconnections',
140                            'oauthapps', 'email', 'sms', 'userdesign', 'other') as $s) {
141                 $m->connect('settings/'.$s, array('action' => $s.'settings'));
142             }
143
144             $m->connect('settings/oauthapps/show/:id',
145                 array('action' => 'showapplication'),
146                 array('id' => '[0-9]+')
147             );
148             $m->connect('settings/oauthapps/new',
149                 array('action' => 'newapplication')
150             );
151             $m->connect('settings/oauthapps/edit/:id',
152                 array('action' => 'editapplication'),
153                 array('id' => '[0-9]+')
154             );
155
156             // search
157
158             foreach (array('group', 'people', 'notice') as $s) {
159                 $m->connect('search/'.$s, array('action' => $s.'search'));
160                 $m->connect('search/'.$s.'?q=:q',
161                             array('action' => $s.'search'),
162                             array('q' => '.+'));
163             }
164
165             // The second of these is needed to make the link work correctly
166             // when inserted into the page. The first is needed to match the
167             // route on the way in. Seems to be another Net_URL_Mapper bug to me.
168             $m->connect('search/notice/rss', array('action' => 'noticesearchrss'));
169             $m->connect('search/notice/rss?q=:q', array('action' => 'noticesearchrss'),
170                         array('q' => '.+'));
171
172             $m->connect('attachment/:attachment',
173                         array('action' => 'attachment'),
174                         array('attachment' => '[0-9]+'));
175
176             $m->connect('attachment/:attachment/ajax',
177                         array('action' => 'attachment_ajax'),
178                         array('attachment' => '[0-9]+'));
179
180             $m->connect('attachment/:attachment/thumbnail',
181                         array('action' => 'attachment_thumbnail'),
182                         array('attachment' => '[0-9]+'));
183
184             $m->connect('notice/new', array('action' => 'newnotice'));
185             $m->connect('notice/new?replyto=:replyto',
186                         array('action' => 'newnotice'),
187                         array('replyto' => '[A-Za-z0-9_-]+'));
188             $m->connect('notice/new?replyto=:replyto&inreplyto=:inreplyto',
189                         array('action' => 'newnotice'),
190                         array('replyto' => '[A-Za-z0-9_-]+'),
191                         array('inreplyto' => '[0-9]+'));
192
193             $m->connect('notice/:notice/file',
194                         array('action' => 'file'),
195                         array('notice' => '[0-9]+'));
196
197             $m->connect('notice/:notice',
198                         array('action' => 'shownotice'),
199                         array('notice' => '[0-9]+'));
200             $m->connect('notice/delete', array('action' => 'deletenotice'));
201             $m->connect('notice/delete/:notice',
202                         array('action' => 'deletenotice'),
203                         array('notice' => '[0-9]+'));
204
205             $m->connect('bookmarklet/new', array('action' => 'bookmarklet'));
206
207             // conversation
208
209             $m->connect('conversation/:id',
210                         array('action' => 'conversation'),
211                         array('id' => '[0-9]+'));
212
213             $m->connect('message/new', array('action' => 'newmessage'));
214             $m->connect('message/new?to=:to', array('action' => 'newmessage'), array('to' => '[A-Za-z0-9_-]+'));
215             $m->connect('message/:message',
216                         array('action' => 'showmessage'),
217                         array('message' => '[0-9]+'));
218
219             $m->connect('user/:id',
220                         array('action' => 'userbyid'),
221                         array('id' => '[0-9]+'));
222
223             $m->connect('tags/', array('action' => 'publictagcloud'));
224             $m->connect('tag/', array('action' => 'publictagcloud'));
225             $m->connect('tags', array('action' => 'publictagcloud'));
226             $m->connect('tag', array('action' => 'publictagcloud'));
227             $m->connect('tag/:tag/rss',
228                         array('action' => 'tagrss'),
229                         array('tag' => '[a-zA-Z0-9]+'));
230             $m->connect('tag/:tag',
231                         array('action' => 'tag'),
232                         array('tag' => '[\pL\pN_\-\.]{1,64}'));
233
234             $m->connect('peopletag/:tag',
235                         array('action' => 'peopletag'),
236                         array('tag' => '[a-zA-Z0-9]+'));
237
238             // groups
239
240             $m->connect('group/new', array('action' => 'newgroup'));
241
242             foreach (array('edit', 'join', 'leave') as $v) {
243                 $m->connect('group/:nickname/'.$v,
244                             array('action' => $v.'group'),
245                             array('nickname' => '[a-zA-Z0-9]+'));
246             }
247
248             foreach (array('members', 'logo', 'rss', 'designsettings') as $n) {
249                 $m->connect('group/:nickname/'.$n,
250                             array('action' => 'group'.$n),
251                             array('nickname' => '[a-zA-Z0-9]+'));
252             }
253
254             $m->connect('group/:nickname/foaf',
255                         array('action' => 'foafgroup'),
256                         array('nickname' => '[a-zA-Z0-9]+'));
257
258             $m->connect('group/:nickname/blocked',
259                         array('action' => 'blockedfromgroup'),
260                         array('nickname' => '[a-zA-Z0-9]+'));
261
262             $m->connect('group/:nickname/makeadmin',
263                         array('action' => 'makeadmin'),
264                         array('nickname' => '[a-zA-Z0-9]+'));
265
266             $m->connect('group/:id/id',
267                         array('action' => 'groupbyid'),
268                         array('id' => '[0-9]+'));
269
270             $m->connect('group/:nickname',
271                         array('action' => 'showgroup'),
272                         array('nickname' => '[a-zA-Z0-9]+'));
273
274             $m->connect('group/', array('action' => 'groups'));
275             $m->connect('group', array('action' => 'groups'));
276             $m->connect('groups/', array('action' => 'groups'));
277             $m->connect('groups', array('action' => 'groups'));
278
279             // Twitter-compatible API
280
281             // statuses API
282
283             $m->connect('api/statuses/public_timeline.:format',
284                         array('action' => 'ApiTimelinePublic',
285                               'format' => '(xml|json|rss|atom)'));
286
287             $m->connect('api/statuses/friends_timeline.:format',
288                         array('action' => 'ApiTimelineFriends',
289                               'format' => '(xml|json|rss|atom)'));
290
291             $m->connect('api/statuses/friends_timeline/:id.:format',
292                         array('action' => 'ApiTimelineFriends',
293                               'id' => '[a-zA-Z0-9]+',
294                               'format' => '(xml|json|rss|atom)'));
295
296             $m->connect('api/statuses/home_timeline.:format',
297                         array('action' => 'ApiTimelineHome',
298                               'format' => '(xml|json|rss|atom)'));
299
300             $m->connect('api/statuses/home_timeline/:id.:format',
301                         array('action' => 'ApiTimelineHome',
302                               'id' => '[a-zA-Z0-9]+',
303                               'format' => '(xml|json|rss|atom)'));
304
305             $m->connect('api/statuses/user_timeline.:format',
306                         array('action' => 'ApiTimelineUser',
307                               'format' => '(xml|json|rss|atom)'));
308
309             $m->connect('api/statuses/user_timeline/:id.:format',
310                         array('action' => 'ApiTimelineUser',
311                               'id' => '[a-zA-Z0-9]+',
312                               'format' => '(xml|json|rss|atom)'));
313
314             $m->connect('api/statuses/mentions.:format',
315                         array('action' => 'ApiTimelineMentions',
316                               'format' => '(xml|json|rss|atom)'));
317
318             $m->connect('api/statuses/mentions/:id.:format',
319                         array('action' => 'ApiTimelineMentions',
320                               'id' => '[a-zA-Z0-9]+',
321                               'format' => '(xml|json|rss|atom)'));
322
323             $m->connect('api/statuses/replies.:format',
324                         array('action' => 'ApiTimelineMentions',
325                               'format' => '(xml|json|rss|atom)'));
326
327             $m->connect('api/statuses/replies/:id.:format',
328                         array('action' => 'ApiTimelineMentions',
329                               'id' => '[a-zA-Z0-9]+',
330                               'format' => '(xml|json|rss|atom)'));
331
332             $m->connect('api/statuses/retweeted_by_me.:format',
333                         array('action' => 'ApiTimelineRetweetedByMe',
334                               'format' => '(xml|json|atom)'));
335
336             $m->connect('api/statuses/retweeted_to_me.:format',
337                         array('action' => 'ApiTimelineRetweetedToMe',
338                               'format' => '(xml|json|atom)'));
339
340             $m->connect('api/statuses/retweets_of_me.:format',
341                         array('action' => 'ApiTimelineRetweetsOfMe',
342                               'format' => '(xml|json|atom)'));
343
344             $m->connect('api/statuses/friends.:format',
345                         array('action' => 'ApiUserFriends',
346                               'format' => '(xml|json)'));
347
348             $m->connect('api/statuses/friends/:id.:format',
349                         array('action' => 'ApiUserFriends',
350                               'id' => '[a-zA-Z0-9]+',
351                               'format' => '(xml|json)'));
352
353             $m->connect('api/statuses/followers.:format',
354                         array('action' => 'ApiUserFollowers',
355                               'format' => '(xml|json)'));
356
357             $m->connect('api/statuses/followers/:id.:format',
358                         array('action' => 'ApiUserFollowers',
359                               'id' => '[a-zA-Z0-9]+',
360                               'format' => '(xml|json)'));
361
362             $m->connect('api/statuses/show.:format',
363                         array('action' => 'ApiStatusesShow',
364                               'format' => '(xml|json)'));
365
366             $m->connect('api/statuses/show/:id.:format',
367                         array('action' => 'ApiStatusesShow',
368                               'id' => '[0-9]+',
369                               'format' => '(xml|json)'));
370
371             $m->connect('api/statuses/update.:format',
372                         array('action' => 'ApiStatusesUpdate',
373                               'format' => '(xml|json)'));
374
375             $m->connect('api/statuses/destroy.:format',
376                         array('action' => 'ApiStatusesDestroy',
377                               'format' => '(xml|json)'));
378
379             $m->connect('api/statuses/destroy/:id.:format',
380                         array('action' => 'ApiStatusesDestroy',
381                               'id' => '[0-9]+',
382                               'format' => '(xml|json)'));
383
384             $m->connect('api/statuses/retweet/:id.:format',
385                         array('action' => 'ApiStatusesRetweet',
386                               'id' => '[0-9]+',
387                               'format' => '(xml|json)'));
388
389             $m->connect('api/statuses/retweets/:id.:format',
390                         array('action' => 'ApiStatusesRetweets',
391                               'id' => '[0-9]+',
392                               'format' => '(xml|json)'));
393
394             // users
395
396             $m->connect('api/users/show.:format',
397                         array('action' => 'ApiUserShow',
398                               'format' => '(xml|json)'));
399
400             $m->connect('api/users/show/:id.:format',
401                         array('action' => 'ApiUserShow',
402                               'id' => '[a-zA-Z0-9]+',
403                               'format' => '(xml|json)'));
404
405             // direct messages
406
407             $m->connect('api/direct_messages.:format',
408                         array('action' => 'ApiDirectMessage',
409                               'format' => '(xml|json|rss|atom)'));
410
411             $m->connect('api/direct_messages/sent.:format',
412                         array('action' => 'ApiDirectMessage',
413                               'format' => '(xml|json|rss|atom)',
414                               'sent' => true));
415
416             $m->connect('api/direct_messages/new.:format',
417                         array('action' => 'ApiDirectMessageNew',
418                               'format' => '(xml|json)'));
419
420             // friendships
421
422             $m->connect('api/friendships/show.:format',
423                         array('action' => 'ApiFriendshipsShow',
424                               'format' => '(xml|json)'));
425
426             $m->connect('api/friendships/exists.:format',
427                         array('action' => 'ApiFriendshipsExists',
428                               'format' => '(xml|json)'));
429
430             $m->connect('api/friendships/create.:format',
431                         array('action' => 'ApiFriendshipsCreate',
432                               'format' => '(xml|json)'));
433
434             $m->connect('api/friendships/destroy.:format',
435                         array('action' => 'ApiFriendshipsDestroy',
436                               'format' => '(xml|json)'));
437
438             $m->connect('api/friendships/create/:id.:format',
439                         array('action' => 'ApiFriendshipsCreate',
440                               'id' => '[a-zA-Z0-9]+',
441                               'format' => '(xml|json)'));
442
443             $m->connect('api/friendships/destroy/:id.:format',
444                         array('action' => 'ApiFriendshipsDestroy',
445                               'id' => '[a-zA-Z0-9]+',
446                               'format' => '(xml|json)'));
447
448             // Social graph
449
450             $m->connect('api/friends/ids/:id.:format',
451                         array('action' => 'apiuserfriends',
452                               'ids_only' => true));
453
454             $m->connect('api/followers/ids/:id.:format',
455                         array('action' => 'apiuserfollowers',
456                               'ids_only' => true));
457
458             $m->connect('api/friends/ids.:format',
459                         array('action' => 'apiuserfriends',
460                               'ids_only' => true));
461
462             $m->connect('api/followers/ids.:format',
463                         array('action' => 'apiuserfollowers',
464                               'ids_only' => true));
465
466             // account
467
468             $m->connect('api/account/verify_credentials.:format',
469                         array('action' => 'ApiAccountVerifyCredentials'));
470
471             $m->connect('api/account/update_profile.:format',
472                         array('action' => 'ApiAccountUpdateProfile'));
473
474             $m->connect('api/account/update_profile_image.:format',
475                         array('action' => 'ApiAccountUpdateProfileImage'));
476
477             $m->connect('api/account/update_profile_background_image.:format',
478                         array('action' => 'ApiAccountUpdateProfileBackgroundImage'));
479
480             $m->connect('api/account/update_profile_colors.:format',
481                         array('action' => 'ApiAccountUpdateProfileColors'));
482
483             $m->connect('api/account/update_delivery_device.:format',
484                         array('action' => 'ApiAccountUpdateDeliveryDevice'));
485
486             // special case where verify_credentials is called w/out a format
487
488             $m->connect('api/account/verify_credentials',
489                         array('action' => 'ApiAccountVerifyCredentials'));
490
491             $m->connect('api/account/rate_limit_status.:format',
492                         array('action' => 'ApiAccountRateLimitStatus'));
493
494             // favorites
495
496             $m->connect('api/favorites.:format',
497                         array('action' => 'ApiTimelineFavorites',
498                               'format' => '(xml|json|rss|atom)'));
499
500             $m->connect('api/favorites/:id.:format',
501                         array('action' => 'ApiTimelineFavorites',
502                               'id' => '[a-zA-Z0-9]+',
503                               'format' => '(xmljson|rss|atom)'));
504
505             $m->connect('api/favorites/create/:id.:format',
506                         array('action' => 'ApiFavoriteCreate',
507                               'id' => '[a-zA-Z0-9]+',
508                               'format' => '(xml|json)'));
509
510             $m->connect('api/favorites/destroy/:id.:format',
511                         array('action' => 'ApiFavoriteDestroy',
512                               'id' => '[a-zA-Z0-9]+',
513                               'format' => '(xml|json)'));
514             // blocks
515
516             $m->connect('api/blocks/create/:id.:format',
517                         array('action' => 'ApiBlockCreate',
518                               'id' => '[a-zA-Z0-9]+',
519                               'format' => '(xml|json)'));
520
521             $m->connect('api/blocks/destroy/:id.:format',
522                         array('action' => 'ApiBlockDestroy',
523                               'id' => '[a-zA-Z0-9]+',
524                               'format' => '(xml|json)'));
525             // help
526
527             $m->connect('api/help/test.:format',
528                         array('action' => 'ApiHelpTest',
529                               'format' => '(xml|json)'));
530
531             // statusnet
532
533             $m->connect('api/statusnet/version.:format',
534                         array('action' => 'ApiStatusnetVersion',
535                               'format' => '(xml|json)'));
536
537             $m->connect('api/statusnet/config.:format',
538                         array('action' => 'ApiStatusnetConfig',
539                               'format' => '(xml|json)'));
540
541             // For older methods, we provide "laconica" base action
542
543             $m->connect('api/laconica/version.:format',
544                         array('action' => 'ApiStatusnetVersion',
545                               'format' => '(xml|json)'));
546
547             $m->connect('api/laconica/config.:format',
548                         array('action' => 'ApiStatusnetConfig',
549                               'format' => '(xml|json)'));
550
551             // Groups and tags are newer than 0.8.1 so no backward-compatibility
552             // necessary
553
554             // Groups
555             //'list' has to be handled differently, as php will not allow a method to be named 'list'
556
557             $m->connect('api/statusnet/groups/timeline/:id.:format',
558                         array('action' => 'ApiTimelineGroup',
559                               'id' => '[a-zA-Z0-9]+',
560                               'format' => '(xmljson|rss|atom)'));
561
562             $m->connect('api/statusnet/groups/show.:format',
563                         array('action' => 'ApiGroupShow',
564                               'format' => '(xml|json)'));
565
566             $m->connect('api/statusnet/groups/show/:id.:format',
567                         array('action' => 'ApiGroupShow',
568                               'id' => '[a-zA-Z0-9]+',
569                               'format' => '(xml|json)'));
570
571             $m->connect('api/statusnet/groups/join.:format',
572                         array('action' => 'ApiGroupJoin',
573                               'id' => '[a-zA-Z0-9]+',
574                               'format' => '(xml|json)'));
575
576             $m->connect('api/statusnet/groups/join/:id.:format',
577                         array('action' => 'ApiGroupJoin',
578                               'format' => '(xml|json)'));
579
580             $m->connect('api/statusnet/groups/leave.:format',
581                         array('action' => 'ApiGroupLeave',
582                               'id' => '[a-zA-Z0-9]+',
583                               'format' => '(xml|json)'));
584
585             $m->connect('api/statusnet/groups/leave/:id.:format',
586                         array('action' => 'ApiGroupLeave',
587                               'format' => '(xml|json)'));
588
589             $m->connect('api/statusnet/groups/is_member.:format',
590                         array('action' => 'ApiGroupIsMember',
591                               'format' => '(xml|json)'));
592
593             $m->connect('api/statusnet/groups/list.:format',
594                         array('action' => 'ApiGroupList',
595                               'format' => '(xml|json|rss|atom)'));
596
597             $m->connect('api/statusnet/groups/list/:id.:format',
598                         array('action' => 'ApiGroupList',
599                               'id' => '[a-zA-Z0-9]+',
600                               'format' => '(xml|json|rss|atom)'));
601
602             $m->connect('api/statusnet/groups/list_all.:format',
603                         array('action' => 'ApiGroupListAll',
604                               'format' => '(xml|json|rss|atom)'));
605
606             $m->connect('api/statusnet/groups/membership.:format',
607                         array('action' => 'ApiGroupMembership',
608                               'format' => '(xml|json)'));
609
610             $m->connect('api/statusnet/groups/membership/:id.:format',
611                         array('action' => 'ApiGroupMembership',
612                               'id' => '[a-zA-Z0-9]+',
613                               'format' => '(xml|json)'));
614
615             $m->connect('api/statusnet/groups/create.:format',
616                         array('action' => 'ApiGroupCreate',
617                               'format' => '(xml|json)'));
618             // Tags
619             $m->connect('api/statusnet/tags/timeline/:tag.:format',
620                         array('action' => 'ApiTimelineTag',
621                               'format' => '(xmljson|rss|atom)'));
622
623             // search
624             $m->connect('api/search.atom', array('action' => 'twitapisearchatom'));
625             $m->connect('api/search.json', array('action' => 'twitapisearchjson'));
626             $m->connect('api/trends.json', array('action' => 'twitapitrends'));
627
628             $m->connect('api/oauth/request_token',
629                         array('action' => 'apioauthrequesttoken'));
630
631             $m->connect('api/oauth/access_token',
632                         array('action' => 'apioauthaccesstoken'));
633
634             $m->connect('api/oauth/authorize',
635                         array('action' => 'apioauthauthorize'));
636
637             // Admin
638
639             $m->connect('admin/site', array('action' => 'siteadminpanel'));
640             $m->connect('admin/design', array('action' => 'designadminpanel'));
641             $m->connect('admin/user', array('action' => 'useradminpanel'));
642                 $m->connect('admin/access', array('action' => 'accessadminpanel'));
643             $m->connect('admin/paths', array('action' => 'pathsadminpanel'));
644             $m->connect('admin/sessions', array('action' => 'sessionsadminpanel'));
645
646             $m->connect('getfile/:filename',
647                         array('action' => 'getfile'),
648                         array('filename' => '[A-Za-z0-9._-]+'));
649
650             // In the "root"
651
652             if (common_config('singleuser', 'enabled')) {
653
654                 $user = User::siteOwner();
655
656                 if (!empty($user)) {
657                     $nickname = $user->nickname;
658                 } else {
659                     $nickname = common_config('singleuser', 'nickname');
660                     if (empty($nickname)) {
661                         throw new ServerException(_("No single user defined for single-user mode."));
662                     }
663                 }
664
665                 foreach (array('subscriptions', 'subscribers',
666                                'all', 'foaf', 'xrds',
667                                'replies', 'microsummary') as $a) {
668                     $m->connect($a,
669                                 array('action' => $a,
670                                       'nickname' => $nickname));
671                 }
672
673                 foreach (array('subscriptions', 'subscribers') as $a) {
674                     $m->connect($a.'/:tag',
675                                 array('action' => $a,
676                                       'nickname' => $nickname),
677                                 array('tag' => '[a-zA-Z0-9]+'));
678                 }
679
680                 foreach (array('rss', 'groups') as $a) {
681                     $m->connect($a,
682                                 array('action' => 'user'.$a,
683                                       'nickname' => $nickname));
684                 }
685
686                 foreach (array('all', 'replies', 'favorites') as $a) {
687                     $m->connect($a.'/rss',
688                                 array('action' => $a.'rss',
689                                       'nickname' => $nickname));
690                 }
691
692                 $m->connect('favorites',
693                             array('action' => 'showfavorites',
694                                   'nickname' => $nickname));
695
696                 $m->connect('avatar/:size',
697                             array('action' => 'avatarbynickname',
698                                   'nickname' => $nickname),
699                             array('size' => '(original|96|48|24)'));
700
701                 $m->connect('tag/:tag/rss',
702                             array('action' => 'userrss',
703                                   'nickname' => $nickname),
704                             array('tag' => '[a-zA-Z0-9]+'));
705
706                 $m->connect('tag/:tag',
707                             array('action' => 'showstream',
708                                   'nickname' => $nickname),
709                             array('tag' => '[a-zA-Z0-9]+'));
710
711                 $m->connect('',
712                             array('action' => 'showstream',
713                                   'nickname' => $nickname));
714
715             } else {
716
717                 $m->connect('', array('action' => 'public'));
718                 $m->connect('rss', array('action' => 'publicrss'));
719                 $m->connect('featuredrss', array('action' => 'featuredrss'));
720                 $m->connect('favoritedrss', array('action' => 'favoritedrss'));
721                 $m->connect('featured/', array('action' => 'featured'));
722                 $m->connect('featured', array('action' => 'featured'));
723                 $m->connect('favorited/', array('action' => 'favorited'));
724                 $m->connect('favorited', array('action' => 'favorited'));
725
726                 foreach (array('subscriptions', 'subscribers',
727                                'nudge', 'all', 'foaf', 'xrds',
728                                'replies', 'inbox', 'outbox', 'microsummary') as $a) {
729                     $m->connect(':nickname/'.$a,
730                                 array('action' => $a),
731                                 array('nickname' => '[a-zA-Z0-9]{1,64}'));
732                 }
733
734                 foreach (array('subscriptions', 'subscribers') as $a) {
735                     $m->connect(':nickname/'.$a.'/:tag',
736                                 array('action' => $a),
737                                 array('tag' => '[a-zA-Z0-9]+',
738                                       'nickname' => '[a-zA-Z0-9]{1,64}'));
739                 }
740
741                 foreach (array('rss', 'groups') as $a) {
742                     $m->connect(':nickname/'.$a,
743                                 array('action' => 'user'.$a),
744                                 array('nickname' => '[a-zA-Z0-9]{1,64}'));
745                 }
746
747                 foreach (array('all', 'replies', 'favorites') as $a) {
748                     $m->connect(':nickname/'.$a.'/rss',
749                                 array('action' => $a.'rss'),
750                                 array('nickname' => '[a-zA-Z0-9]{1,64}'));
751                 }
752
753                 $m->connect(':nickname/favorites',
754                             array('action' => 'showfavorites'),
755                             array('nickname' => '[a-zA-Z0-9]{1,64}'));
756
757                 $m->connect(':nickname/avatar/:size',
758                             array('action' => 'avatarbynickname'),
759                             array('size' => '(original|96|48|24)',
760                                   'nickname' => '[a-zA-Z0-9]{1,64}'));
761
762                 $m->connect(':nickname/tag/:tag/rss',
763                             array('action' => 'userrss'),
764                             array('nickname' => '[a-zA-Z0-9]{1,64}'),
765                             array('tag' => '[a-zA-Z0-9]+'));
766
767                 $m->connect(':nickname/tag/:tag',
768                             array('action' => 'showstream'),
769                             array('nickname' => '[a-zA-Z0-9]{1,64}'),
770                             array('tag' => '[a-zA-Z0-9]+'));
771
772                 $m->connect(':nickname',
773                             array('action' => 'showstream'),
774                             array('nickname' => '[a-zA-Z0-9]{1,64}'));
775             }
776
777             // user stuff
778
779             Event::handle('RouterInitialized', array($m));
780         }
781
782         return $m;
783     }
784
785     function map($path)
786     {
787         try {
788             $match = $this->m->match($path);
789         } catch (Net_URL_Mapper_InvalidException $e) {
790             common_log(LOG_ERR, "Problem getting route for $path - " .
791                        $e->getMessage());
792             $cac = new ClientErrorAction("Page not found.", 404);
793             $cac->showPage();
794         }
795
796         return $match;
797     }
798
799     function build($action, $args=null, $params=null, $fragment=null)
800     {
801         $action_arg = array('action' => $action);
802
803         if ($args) {
804             $args = array_merge($action_arg, $args);
805         } else {
806             $args = $action_arg;
807         }
808
809         $url = $this->m->generate($args, $params, $fragment);
810
811         // Due to a bug in the Net_URL_Mapper code, the returned URL may
812         // contain a malformed query of the form ?p1=v1?p2=v2?p3=v3. We
813         // repair that here rather than modifying the upstream code...
814
815         $qpos = strpos($url, '?');
816         if ($qpos !== false) {
817             $url = substr($url, 0, $qpos+1) .
818               str_replace('?', '&', substr($url, $qpos+1));
819         }
820         return $url;
821     }
822 }