chore: Add additional missing events (#1639)

* chore: Add additional missing events
signed in
profile updated
team setting updated

* Minor refactor to DRY existing code

* Add events

* lint

* flow: Add missing ip to event types
This commit is contained in:
Tom Moor 2020-11-14 20:44:18 -08:00 committed by GitHub
parent 7bdcba46b8
commit 19ab32f551
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 98 additions and 66 deletions

View File

@ -1,7 +1,7 @@
// @flow // @flow
import Router from "koa-router"; import Router from "koa-router";
import auth from "../middlewares/authentication"; import auth from "../middlewares/authentication";
import { Team } from "../models"; import { Event, Team } from "../models";
import policy from "../policies"; import policy from "../policies";
import { presentTeam, presentPolicies } from "../presenters"; import { presentTeam, presentPolicies } from "../presenters";
@ -31,8 +31,24 @@ router.post("team.update", auth(), async (ctx) => {
if (documentEmbeds !== undefined) team.documentEmbeds = documentEmbeds; if (documentEmbeds !== undefined) team.documentEmbeds = documentEmbeds;
if (guestSignin !== undefined) team.guestSignin = guestSignin; if (guestSignin !== undefined) team.guestSignin = guestSignin;
if (avatarUrl !== undefined) team.avatarUrl = avatarUrl; if (avatarUrl !== undefined) team.avatarUrl = avatarUrl;
const changes = team.changed();
const data = {};
await team.save(); await team.save();
for (const change of changes) {
data[change] = team[change];
}
await Event.create({
name: "teams.update",
actorId: user.id,
teamId: user.teamId,
data,
ip: ctx.request.ip,
});
ctx.body = { ctx.body = {
data: presentTeam(team), data: presentTeam(team),
policies: presentPolicies(user, [team]), policies: presentPolicies(user, [team]),

View File

@ -69,6 +69,14 @@ router.post("users.update", auth(), async (ctx) => {
await user.save(); await user.save();
await Event.create({
name: "users.update",
actorId: user.id,
userId: user.id,
teamId: user.teamId,
ip: ctx.request.ip,
});
ctx.body = { ctx.body = {
data: presentUser(user, { includeDetails: true }), data: presentUser(user, { includeDetails: true }),
}; };

View File

@ -5,7 +5,7 @@ import Router from "koa-router";
import { capitalize } from "lodash"; import { capitalize } from "lodash";
import Sequelize from "sequelize"; import Sequelize from "sequelize";
import auth from "../middlewares/authentication"; import auth from "../middlewares/authentication";
import { User, Team, Event } from "../models"; import { User, Team } from "../models";
const Op = Sequelize.Op; const Op = Sequelize.Op;
@ -122,20 +122,6 @@ router.get("google.callback", auth({ required: false }), async (ctx) => {
await team.provisionSubdomain(hostname); await team.provisionSubdomain(hostname);
} }
if (isFirstSignin) {
await Event.create({
name: "users.create",
actorId: user.id,
userId: user.id,
teamId: team.id,
data: {
name: user.name,
service: "google",
},
ip: ctx.request.ip,
});
}
// set cookies on response and redirect to team subdomain // set cookies on response and redirect to team subdomain
ctx.signIn(user, team, "google", isFirstSignin); ctx.signIn(user, team, "google", isFirstSignin);
} catch (err) { } catch (err) {

View File

@ -4,14 +4,7 @@ import Router from "koa-router";
import Sequelize from "sequelize"; import Sequelize from "sequelize";
import { slackAuth } from "../../shared/utils/routeHelpers"; import { slackAuth } from "../../shared/utils/routeHelpers";
import auth from "../middlewares/authentication"; import auth from "../middlewares/authentication";
import { import { Authentication, Collection, Integration, User, Team } from "../models";
Authentication,
Collection,
Integration,
User,
Event,
Team,
} from "../models";
import * as Slack from "../slack"; import * as Slack from "../slack";
import { getCookieDomain } from "../utils/domains"; import { getCookieDomain } from "../utils/domains";
@ -101,20 +94,6 @@ router.get("slack.callback", auth({ required: false }), async (ctx) => {
await team.provisionSubdomain(data.team.domain); await team.provisionSubdomain(data.team.domain);
} }
if (isFirstSignin) {
await Event.create({
name: "users.create",
actorId: user.id,
userId: user.id,
teamId: team.id,
data: {
name: user.name,
service: "slack",
},
ip: ctx.request.ip,
});
}
// set cookies on response and redirect to team subdomain // set cookies on response and redirect to team subdomain
ctx.signIn(user, team, "slack", isFirstSignin); ctx.signIn(user, team, "slack", isFirstSignin);
} catch (err) { } catch (err) {

View File

@ -9,6 +9,7 @@ const log = debug("services");
export type UserEvent = export type UserEvent =
| { | {
name: | "users.create" // eslint-disable-line name: | "users.create" // eslint-disable-line
| "users.signin"
| "users.update" | "users.update"
| "users.suspend" | "users.suspend"
| "users.activate" | "users.activate"
@ -16,6 +17,7 @@ export type UserEvent =
userId: string, userId: string,
teamId: string, teamId: string,
actorId: string, actorId: string,
ip: string,
} }
| { | {
name: "users.invite", name: "users.invite",
@ -25,6 +27,7 @@ export type UserEvent =
email: string, email: string,
name: string, name: string,
}, },
ip: string,
}; };
export type DocumentEvent = export type DocumentEvent =
@ -43,6 +46,7 @@ export type DocumentEvent =
collectionId: string, collectionId: string,
teamId: string, teamId: string,
actorId: string, actorId: string,
ip: string,
} }
| { | {
name: "documents.move", name: "documents.move",
@ -54,6 +58,7 @@ export type DocumentEvent =
collectionIds: string[], collectionIds: string[],
documentIds: string[], documentIds: string[],
}, },
ip: string,
} }
| { | {
name: | "documents.update" // eslint-disable-line name: | "documents.update" // eslint-disable-line
@ -69,6 +74,7 @@ export type DocumentEvent =
autosave: boolean, autosave: boolean,
done: boolean, done: boolean,
}, },
ip: string,
} }
| { | {
name: "documents.title_change", name: "documents.title_change",
@ -81,6 +87,7 @@ export type DocumentEvent =
title: string, title: string,
previousTitle: string, previousTitle: string,
}, },
ip: string,
}; };
export type RevisionEvent = { export type RevisionEvent = {
@ -98,6 +105,7 @@ export type CollectionEvent =
collectionId: string, collectionId: string,
teamId: string, teamId: string,
actorId: string, actorId: string,
ip: string,
} }
| { | {
name: "collections.add_user" | "collections.remove_user", name: "collections.add_user" | "collections.remove_user",
@ -105,6 +113,7 @@ export type CollectionEvent =
collectionId: string, collectionId: string,
teamId: string, teamId: string,
actorId: string, actorId: string,
ip: string,
} }
| { | {
name: "collections.add_group" | "collections.remove_group", name: "collections.add_group" | "collections.remove_group",
@ -139,6 +148,15 @@ export type IntegrationEvent = {
modelId: string, modelId: string,
teamId: string, teamId: string,
actorId: string, actorId: string,
ip: string,
};
export type TeamEvent = {
name: "teams.update",
teamId: string,
actorId: string,
data: Object,
ip: string,
}; };
export type Event = export type Event =
@ -147,7 +165,8 @@ export type Event =
| CollectionEvent | CollectionEvent
| IntegrationEvent | IntegrationEvent
| GroupEvent | GroupEvent
| RevisionEvent; | RevisionEvent
| TeamEvent;
const globalEventsQueue = createQueue("global events"); const globalEventsQueue = createQueue("global events");
const serviceEventsQueue = createQueue("service events"); const serviceEventsQueue = createQueue("service events");

View File

@ -2,7 +2,7 @@
import addMonths from "date-fns/add_months"; import addMonths from "date-fns/add_months";
import JWT from "jsonwebtoken"; import JWT from "jsonwebtoken";
import { AuthenticationError, UserSuspendedError } from "../errors"; import { AuthenticationError, UserSuspendedError } from "../errors";
import { User, Team, ApiKey } from "../models"; import { User, Event, Team, ApiKey } from "../models";
import type { ContextWithState } from "../types"; import type { ContextWithState } from "../types";
import { getCookieDomain } from "../utils/domains"; import { getCookieDomain } from "../utils/domains";
import { getUserForJWT } from "../utils/jwt"; import { getUserForJWT } from "../utils/jwt";
@ -94,12 +94,7 @@ export default function auth(options?: { required?: boolean } = {}) {
ctx.state.user = user; ctx.state.user = user;
} }
ctx.signIn = async ( ctx.signIn = (user: User, team: Team, service, isFirstSignin = false) => {
user: User,
team: Team,
service,
isFirstSignin = false
) => {
if (user.isSuspended) { if (user.isSuspended) {
return ctx.redirect("/?notice=suspended"); return ctx.redirect("/?notice=suspended");
} }
@ -107,6 +102,32 @@ export default function auth(options?: { required?: boolean } = {}) {
// update the database when the user last signed in // update the database when the user last signed in
user.updateSignedIn(ctx.request.ip); user.updateSignedIn(ctx.request.ip);
if (isFirstSignin) {
Event.create({
name: "users.create",
actorId: user.id,
userId: user.id,
teamId: team.id,
data: {
name: user.name,
service,
},
ip: ctx.request.ip,
});
} else {
Event.create({
name: "users.signin",
actorId: user.id,
userId: user.id,
teamId: team.id,
data: {
name: user.name,
service,
},
ip: ctx.request.ip,
});
}
const domain = getCookieDomain(ctx.request.hostname); const domain = getCookieDomain(ctx.request.hostname);
const expires = addMonths(new Date(), 3); const expires = addMonths(new Date(), 3);

View File

@ -55,7 +55,8 @@ Event.add = (event) => {
}; };
Event.ACTIVITY_EVENTS = [ Event.ACTIVITY_EVENTS = [
"users.create", "collections.create",
"collections.delete",
"documents.publish", "documents.publish",
"documents.archive", "documents.archive",
"documents.unarchive", "documents.unarchive",
@ -63,20 +64,19 @@ Event.ACTIVITY_EVENTS = [
"documents.unpin", "documents.unpin",
"documents.delete", "documents.delete",
"documents.restore", "documents.restore",
"collections.create", "users.create",
"collections.delete",
]; ];
Event.AUDIT_EVENTS = [ Event.AUDIT_EVENTS = [
"api_keys.create", "api_keys.create",
"api_keys.delete", "api_keys.delete",
"users.create", "collections.create",
"users.promote", "collections.update",
"users.demote", "collections.add_user",
"users.invite", "collections.remove_user",
"users.suspend", "collections.add_group",
"users.activate", "collections.remove_group",
"users.delete", "collections.delete",
"documents.create", "documents.create",
"documents.publish", "documents.publish",
"documents.update", "documents.update",
@ -86,19 +86,22 @@ Event.AUDIT_EVENTS = [
"documents.unpin", "documents.unpin",
"documents.move", "documents.move",
"documents.delete", "documents.delete",
"shares.create",
"shares.update",
"shares.revoke",
"groups.create", "groups.create",
"groups.update", "groups.update",
"groups.delete", "groups.delete",
"collections.create", "shares.create",
"collections.update", "shares.update",
"collections.add_user", "shares.revoke",
"collections.remove_user", "teams.update",
"collections.add_group", "users.create",
"collections.remove_group", "users.update",
"collections.delete", "users.signin",
"users.promote",
"users.demote",
"users.invite",
"users.suspend",
"users.activate",
"users.delete",
]; ];
export default Event; export default Event;