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:
parent
7bdcba46b8
commit
19ab32f551
|
@ -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]),
|
||||||
|
|
|
@ -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 }),
|
||||||
};
|
};
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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");
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Reference in New Issue