summaryrefslogtreecommitdiff
path: root/src/processor/http/follow.ts
diff options
context:
space:
mode:
authorsyuilo <Syuilotan@yahoo.co.jp>2018-04-01 23:32:46 +0900
committerGitHub <noreply@github.com>2018-04-01 23:32:46 +0900
commit13c2de67a4e5aedc9e70726752c15ff232b7c9fd (patch)
tree3da38725843c01e0dddf9a044e8017018b735325 /src/processor/http/follow.ts
parentAdd missing semicolon (diff)
parentImplement remote follow (diff)
downloadsharkey-13c2de67a4e5aedc9e70726752c15ff232b7c9fd.tar.gz
sharkey-13c2de67a4e5aedc9e70726752c15ff232b7c9fd.tar.bz2
sharkey-13c2de67a4e5aedc9e70726752c15ff232b7c9fd.zip
Merge pull request #1356 from akihikodaki/publickey
Implement remote follow
Diffstat (limited to 'src/processor/http/follow.ts')
-rw-r--r--src/processor/http/follow.ts89
1 files changed, 89 insertions, 0 deletions
diff --git a/src/processor/http/follow.ts b/src/processor/http/follow.ts
new file mode 100644
index 0000000000..adaa2f3f65
--- /dev/null
+++ b/src/processor/http/follow.ts
@@ -0,0 +1,89 @@
+import { request } from 'https';
+import { sign } from 'http-signature';
+import { URL } from 'url';
+import User, { ILocalAccount, IRemoteAccount, pack as packUser } from '../../models/user';
+import Following from '../../models/following';
+import event from '../../common/event';
+import notify from '../../common/notify';
+import context from '../../common/remote/activitypub/renderer/context';
+import render from '../../common/remote/activitypub/renderer/follow';
+import config from '../../conf';
+
+export default ({ data }, done) => Following.findOne({ _id: data.following }).then(({ followerId, followeeId }) => {
+ const promisedFollower = User.findOne({ _id: followerId });
+ const promisedFollowee = User.findOne({ _id: followeeId });
+
+ return Promise.all([
+ // Increment following count
+ User.update(followerId, {
+ $inc: {
+ followingCount: 1
+ }
+ }),
+
+ // Increment followers count
+ User.update({ _id: followeeId }, {
+ $inc: {
+ followersCount: 1
+ }
+ }),
+
+ // Notify
+ promisedFollowee.then(followee => followee.host === null ?
+ notify(followeeId, followerId, 'follow') : null),
+
+ // Publish follow event
+ Promise.all([promisedFollower, promisedFollowee]).then(([follower, followee]) => {
+ const followerEvent = packUser(followee, follower)
+ .then(packed => event(follower._id, 'follow', packed));
+ let followeeEvent;
+
+ if (followee.host === null) {
+ followeeEvent = packUser(follower, followee)
+ .then(packed => event(followee._id, 'followed', packed));
+ } else {
+ followeeEvent = new Promise((resolve, reject) => {
+ const {
+ protocol,
+ hostname,
+ port,
+ pathname,
+ search
+ } = new URL(followee.account as IRemoteAccount).inbox);
+
+ const req = request({
+ protocol,
+ hostname,
+ port,
+ method: 'POST',
+ path: pathname + search,
+ }, res => {
+ res.on('close', () => {
+ if (res.statusCode >= 200 && res.statusCode < 300) {
+ resolve();
+ } else {
+ reject(res);
+ }
+ });
+
+ res.on('data', () => {});
+ res.on('error', reject);
+ });
+
+ sign(req, {
+ authorizationHeaderName: 'Signature',
+ key: (follower.account as ILocalAccount).keypair,
+ keyId: `acct:${follower.username}@${config.host}`
+ });
+
+ const rendered = render(follower, followee);
+ rendered['@context'] = context;
+
+ req.end(JSON.stringify(rendered));
+ });
+ }
+
+ return Promise.all([followerEvent, followeeEvent]);
+ })
+ ]);
+}).then(done, done);