]> git.mxchange.org Git - quix0rs-gnu-social.git/blob - plugins/OStatus/actions/usersalmon.php
Debugging log fix.
[quix0rs-gnu-social.git] / plugins / OStatus / actions / usersalmon.php
1 <?php
2 /*
3  * StatusNet - the distributed open-source microblogging tool
4  * Copyright (C) 2010, StatusNet, Inc.
5  *
6  * This program is free software: you can redistribute it and/or modify
7  * it under the terms of the GNU Affero General Public License as published by
8  * the Free Software Foundation, either version 3 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU Affero General Public License for more details.
15  *
16  * You should have received a copy of the GNU Affero General Public License
17  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
18  */
19
20 if (!defined('GNUSOCIAL')) { exit(1); }
21
22 /**
23  * @package OStatusPlugin
24  * @author James Walker <james@status.net>
25  */
26 class UsersalmonAction extends SalmonAction
27 {
28     protected function prepare(array $args=array())
29     {
30         parent::prepare($args);
31
32         $this->user = User::getByID($this->trimmed('id'));
33
34         $this->target = $this->user->getProfile();
35
36         // Notice must either be a) in reply to a notice by this user
37         // or b) in reply to a notice to the attention of this user
38         // or c) to the attention of this user
39         // or d) reference the user as an activity:object
40
41         $notice = null;
42
43         if (!empty($this->activity->context->replyToID)) {
44             try {
45                 $notice = Notice::getByUri($this->activity->context->replyToID);
46                 common_debug('Referenced Notice object found with URI: '.$notice->getUri());
47             } catch (NoResultException $e) {
48                 common_debug('Referenced Notice object NOT found with URI: '.$this->activity->context->replyToID);
49                 $notice = false;
50             }
51         }
52
53         if ($notice instanceof Notice &&
54                 ($this->target->sameAs($notice->getProfile())
55                     || in_array($this->target->getID(), $notice->getAttentionProfileIDs())
56                 )) {
57             // In reply to a notice either from or mentioning this user.
58             common_debug('User is the owner or was in the attention list of thr:in-reply-to activity.');
59         } elseif (!empty($this->activity->context->attention) &&
60                    array_key_exists($this->target->getUri(), $this->activity->context->attention)) {
61             // To the attention of this user.
62             common_debug('User was in attention list of salmon slap.');
63         } elseif (!empty($this->activity->objects) && $this->activity->objects[0]->id === $this->target->getUri()) {
64             // The user is the object of this slap (unfollow for example)
65             common_debug('User URI was the id of the salmon slap object.');
66         } else {
67             common_debug('User was NOT found in salmon slap context.');
68             // TRANS: Client exception.
69             throw new ClientException(_m('The owner of this salmon endpoint was not in the context of the carried slap.'));
70         }
71
72         return true;
73     }
74
75     /**
76      * We've gotten a post event on the Salmon backchannel, probably a reply.
77      *
78      * @todo validate if we need to handle this post, then call into
79      * ostatus_profile's general incoming-post handling.
80      */
81     function handlePost()
82     {
83         common_log(LOG_INFO, "Received post of '{$this->activity->objects[0]->id}' from '{$this->activity->actor->id}'");
84
85         // @fixme: process all activity objects?
86         switch ($this->activity->objects[0]->type) {
87         case ActivityObject::ARTICLE:
88         case ActivityObject::BLOGENTRY:
89         case ActivityObject::NOTE:
90         case ActivityObject::STATUS:
91         case ActivityObject::COMMENT:
92             break;
93         default:
94             // TRANS: Client exception thrown when an undefied activity is performed.
95             throw new ClientException(_m('Cannot handle that kind of post.'));
96         }
97
98         try {
99             $this->saveNotice();
100         } catch (AlreadyFulfilledException $e) {
101             return;
102         }
103     }
104
105     /**
106      * We've gotten a follow/subscribe notification from a remote user.
107      * Save a subscription relationship for them.
108      */
109     function handleFollow()
110     {
111         common_log(LOG_INFO, sprintf('Setting up subscription from remote %s to local %s', $this->oprofile->getUri(), $this->target->getNickname()));
112         Subscription::start($this->actor, $this->target);
113     }
114
115     /**
116      * We've gotten an unfollow/unsubscribe notification from a remote user.
117      * Check if we have a subscription relationship for them and kill it.
118      *
119      * @fixme probably catch exceptions on fail?
120      */
121     function handleUnfollow()
122     {
123         common_log(LOG_INFO, sprintf('Canceling subscription from remote %s to local %s', $this->oprofile->getUri(), $this->target->getNickname()));
124         try {
125             Subscription::cancel($this->actor, $this->target);
126         } catch (NoProfileException $e) {
127             common_debug('Could not find profile for Subscription: '.$e->getMessage());
128         }
129     }
130
131     function handleTag()
132     {
133         if ($this->activity->target->type == ActivityObject::_LIST) {
134             if ($this->activity->objects[0]->type != ActivityObject::PERSON) {
135                 // TRANS: Client exception.
136                 throw new ClientException(_m('Not a person object.'));
137             }
138             // this is a peopletag
139             $tagged = User::getKV('uri', $this->activity->objects[0]->id);
140
141             if (!$tagged instanceof User) {
142                 // TRANS: Client exception.
143                 throw new ClientException(_m('Unidentified profile being listed.'));
144             }
145
146             if ($tagged->id !== $this->target->id) {
147                 // TRANS: Client exception.
148                 throw new ClientException(_m('This user is not the one being listed.'));
149             }
150
151             // save the list
152             $list   = Ostatus_profile::ensureActivityObjectProfile($this->activity->target);
153
154             $ptag = $list->localPeopletag();
155             $result = Profile_tag::setTag($ptag->tagger, $tagged->id, $ptag->tag);
156             if (!$result) {
157                 // TRANS: Client exception.
158                 throw new ClientException(_m('The listing could not be saved.'));
159             }
160         }
161     }
162
163     function handleUntag()
164     {
165         if ($this->activity->target->type == ActivityObject::_LIST) {
166             if ($this->activity->objects[0]->type != ActivityObject::PERSON) {
167                 // TRANS: Client exception.
168                 throw new ClientException(_m('Not a person object.'));
169             }
170             // this is a peopletag
171             $tagged = User::getKV('uri', $this->activity->objects[0]->id);
172
173             if (!$tagged instanceof User) {
174                 // TRANS: Client exception.
175                 throw new ClientException(_m('Unidentified profile being unlisted.'));
176             }
177
178             if ($tagged->id !== $this->target->id) {
179                 // TRANS: Client exception.
180                 throw new ClientException(_m('This user is not the one being unlisted.'));
181             }
182
183             // save the list
184             $list   = Ostatus_profile::ensureActivityObjectProfile($this->activity->target);
185
186             $ptag = $list->localPeopletag();
187             $result = Profile_tag::unTag($ptag->tagger, $tagged->id, $ptag->tag);
188
189             if (!$result) {
190                 // TRANS: Client exception.
191                 throw new ClientException(_m('The listing could not be deleted.'));
192             }
193         }
194     }
195
196     /**
197      * @param ActivityObject $object
198      * @return Notice
199      * @throws ClientException on invalid input
200      */
201     function getNotice(ActivityObject $object)
202     {
203         switch ($object->type) {
204         case ActivityObject::ARTICLE:
205         case ActivityObject::BLOGENTRY:
206         case ActivityObject::NOTE:
207         case ActivityObject::STATUS:
208         case ActivityObject::COMMENT:
209             break;
210         default:
211             // TRANS: Client exception.
212             throw new ClientException(_m('Cannot handle that kind of object for liking/faving.'));
213         }
214
215         $notice = Notice::getKV('uri', $object->id);
216
217         if (!$notice instanceof Notice) {
218             // TRANS: Client exception. %s is an object ID.
219             throw new ClientException(sprintf(_m('Notice with ID %s unknown.'),$object->id));
220         }
221
222         if ($notice->profile_id != $this->target->id) {
223             // TRANS: Client exception. %1$s is a notice ID, %2$s is a user ID.
224             throw new ClientException(sprintf(_m('Notice with ID %1$s not posted by %2$s.'), $object->id, $this->target->id));
225         }
226
227         return $notice;
228     }
229 }