Passed
Push — master ( fe261c...bd7e98 )
by Morris
18:03 queued 07:36
created

Notifier::__construct()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 12
Code Lines 5

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 1
eloc 5
nc 1
nop 5
dl 0
loc 12
rs 10
c 0
b 0
f 0
1
<?php
2
/**
3
 * @copyright Copyright (c) 2016, ownCloud, Inc.
4
 *
5
 * @author Arthur Schiwon <[email protected]>
6
 * @author Joas Schilling <[email protected]>
7
 * @author Roeland Jago Douma <[email protected]>
8
 *
9
 * @license AGPL-3.0
10
 *
11
 * This code is free software: you can redistribute it and/or modify
12
 * it under the terms of the GNU Affero General Public License, version 3,
13
 * as published by the Free Software Foundation.
14
 *
15
 * This program is distributed in the hope that it will be useful,
16
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18
 * GNU Affero General Public License for more details.
19
 *
20
 * You should have received a copy of the GNU Affero General Public License, version 3,
21
 * along with this program.  If not, see <http://www.gnu.org/licenses/>
22
 *
23
 */
24
25
namespace OCA\Comments\Notification;
26
27
use OCP\Comments\IComment;
28
use OCP\Comments\ICommentsManager;
29
use OCP\Comments\NotFoundException;
30
use OCP\Files\IRootFolder;
31
use OCP\IURLGenerator;
32
use OCP\IUser;
33
use OCP\IUserManager;
34
use OCP\L10N\IFactory;
35
use OCP\Notification\INotification;
36
use OCP\Notification\INotifier;
37
38
class Notifier implements INotifier {
39
40
	/** @var IFactory */
41
	protected $l10nFactory;
42
43
	/** @var IRootFolder  */
44
	protected $rootFolder;
45
46
	/** @var ICommentsManager  */
47
	protected $commentsManager;
48
49
	/** @var IURLGenerator */
50
	protected $url;
51
52
	/** @var IUserManager */
53
	protected $userManager;
54
55
	public function __construct(
56
		IFactory $l10nFactory,
57
		IRootFolder $rootFolder,
58
		ICommentsManager $commentsManager,
59
		IURLGenerator $url,
60
		IUserManager $userManager
61
	) {
62
		$this->l10nFactory = $l10nFactory;
63
		$this->rootFolder = $rootFolder;
64
		$this->commentsManager = $commentsManager;
65
		$this->url = $url;
66
		$this->userManager = $userManager;
67
	}
68
69
	/**
70
	 * @param INotification $notification
71
	 * @param string $languageCode The code of the language that should be used to prepare the notification
72
	 * @return INotification
73
	 * @throws \InvalidArgumentException When the notification was not prepared by a notifier
74
	 */
75
	public function prepare(INotification $notification, $languageCode) {
76
		if($notification->getApp() !== 'comments') {
77
			throw new \InvalidArgumentException();
78
		}
79
		try {
80
			$comment = $this->commentsManager->get($notification->getObjectId());
81
		} catch(NotFoundException $e) {
82
			// needs to be converted to InvalidArgumentException, otherwise none Notifications will be shown at all
83
			throw new \InvalidArgumentException('Comment not found', 0, $e);
84
		}
85
		$l = $this->l10nFactory->get('comments', $languageCode);
86
		$displayName = $comment->getActorId();
87
		$isDeletedActor = $comment->getActorType() === ICommentsManager::DELETED_USER;
88
		if ($comment->getActorType() === 'users') {
89
			$commenter = $this->userManager->get($comment->getActorId());
90
			if ($commenter instanceof IUser) {
91
				$displayName = $commenter->getDisplayName();
92
			}
93
		}
94
95
		switch ($notification->getSubject()) {
96
			case 'mention':
97
				$parameters = $notification->getSubjectParameters();
98
				if($parameters[0] !== 'files') {
99
					throw new \InvalidArgumentException('Unsupported comment object');
100
				}
101
				$userFolder = $this->rootFolder->getUserFolder($notification->getUser());
102
				$nodes = $userFolder->getById((int)$parameters[1]);
103
				if(empty($nodes)) {
104
					throw new \InvalidArgumentException('Cannot resolve file ID to node instance');
105
				}
106
				$node = $nodes[0];
107
108
				$path = rtrim($node->getPath(), '/');
109
				if (strpos($path, '/' . $notification->getUser() . '/files/') === 0) {
110
					// Remove /user/files/...
111
					$fullPath = $path;
112
					list(,,, $path) = explode('/', $fullPath, 4);
113
				}
114
				$subjectParameters = [
115
					'file' => [
116
						'type' => 'file',
117
						'id' => $comment->getObjectId(),
118
						'name' => $node->getName(),
119
						'path' => $path,
120
						'link' => $this->url->linkToRouteAbsolute('files.viewcontroller.showFile', ['fileid' => $comment->getObjectId()]),
121
					],
122
				];
123
124
				if ($isDeletedActor) {
125
					$subject = $l->t('You were mentioned on “{file}”, in a comment by a user that has since been deleted');
126
				} else {
127
					$subject = $l->t('{user} mentioned you in a comment on “{file}”');
128
					$subjectParameters['user'] = [
129
						'type' => 'user',
130
						'id' => $comment->getActorId(),
131
						'name' => $displayName,
132
					];
133
				}
134
				list($message, $messageParameters) = $this->commentToRichMessage($comment);
135
				$notification->setRichSubject($subject, $subjectParameters)
136
					->setParsedSubject($this->richToParsed($subject, $subjectParameters))
137
					->setRichMessage($message, $messageParameters)
138
					->setParsedMessage($this->richToParsed($message, $messageParameters))
139
					->setIcon($this->url->getAbsoluteURL($this->url->imagePath('core', 'actions/comment.svg')))
140
					->setLink($this->url->linkToRouteAbsolute(
141
						'comments.Notifications.view',
142
						['id' => $comment->getId()])
143
					);
144
145
				return $notification;
146
				break;
0 ignored issues
show
Unused Code introduced by
break is not strictly necessary here and could be removed.

The break statement is not necessary if it is preceded for example by a return statement:

switch ($x) {
    case 1:
        return 'foo';
        break; // This break is not necessary and can be left off.
}

If you would like to keep this construct to be consistent with other case statements, you can safely mark this issue as a false-positive.

Loading history...
147
148
			default:
149
				throw new \InvalidArgumentException('Invalid subject');
150
		}
151
	}
152
153
	public function commentToRichMessage(IComment $comment): array {
154
		$message = $comment->getMessage();
155
		$messageParameters = [];
156
		$mentionTypeCount = [];
157
		$mentions = $comment->getMentions();
158
		foreach ($mentions as $mention) {
159
			if ($mention['type'] === 'user') {
160
				$user = $this->userManager->get($mention['id']);
161
				if (!$user instanceof IUser) {
162
					continue;
163
				}
164
			}
165
			if (!array_key_exists($mention['type'], $mentionTypeCount)) {
166
				$mentionTypeCount[$mention['type']] = 0;
167
			}
168
			$mentionTypeCount[$mention['type']]++;
169
			// To keep a limited character set in parameter IDs ([a-zA-Z0-9-])
170
			// the mention parameter ID does not include the mention ID (which
171
			// could contain characters like '@' for user IDs) but a one-based
172
			// index of the mentions of that type.
173
			$mentionParameterId = 'mention-' . $mention['type'] . $mentionTypeCount[$mention['type']];
174
			$message = str_replace('@' . $mention['id'], '{' . $mentionParameterId . '}', $message);
175
			try {
176
				$displayName = $this->commentsManager->resolveDisplayName($mention['type'], $mention['id']);
177
			} catch (\OutOfBoundsException $e) {
178
				// There is no registered display name resolver for the mention
179
				// type, so the client decides what to display.
180
				$displayName = '';
181
			}
182
			$messageParameters[$mentionParameterId] = [
183
				'type' => $mention['type'],
184
				'id' => $mention['id'],
185
				'name' => $displayName
186
			];
187
		}
188
		return [$message, $messageParameters];
189
	}
190
191
	public function richToParsed(string $message, array $parameters): string {
192
		$placeholders = $replacements = [];
193
		foreach ($parameters as $placeholder => $parameter) {
194
			$placeholders[] = '{' . $placeholder . '}';
195
			if ($parameter['type'] === 'user') {
196
				$replacements[] = '@' . $parameter['name'];
197
			} else if ($parameter['type'] === 'file') {
198
				$replacements[] = $parameter['path'];
199
			} else {
200
				$replacements[] = $parameter['name'];
201
			}
202
		}
203
		return str_replace($placeholders, $replacements, $message);
204
	}
205
}
206