-
Notifications
You must be signed in to change notification settings - Fork 66
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #4824 from FlowFuse/invite-reminder
Send invite Reminders
- Loading branch information
Showing
10 changed files
with
273 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
/** | ||
* Add reminderSentAt to Invitations table | ||
*/ | ||
const { DataTypes } = require('sequelize') | ||
|
||
module.exports = { | ||
up: async (context) => { | ||
await context.addColumn('Invitations', 'reminderSentAt', { | ||
type: DataTypes.DATE, | ||
allowNull: true, | ||
defaultValue: null | ||
}) | ||
}, | ||
down: async (context) => {} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
const { Op } = require('sequelize') | ||
|
||
const { randomInt } = require('../utils') | ||
|
||
module.exports = { | ||
name: 'expiredInvites', | ||
startup: true, | ||
// Pick a random hour/minute for this task to run at. If the application is | ||
// horizontal scaled, this will avoid two instances running at the same time | ||
schedule: `${randomInt(0, 59)} ${randomInt(0, 23)} * * *`, | ||
run: async function (app) { | ||
await app.db.models.Invitation.destroy({ where: { expiresAt: { [Op.lt]: Date.now() } } }) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,73 @@ | ||
const { Op } = require('sequelize') | ||
|
||
const { randomInt } = require('../utils') | ||
|
||
module.exports = { | ||
name: 'inviteReminder', | ||
startup: false, | ||
// runs daily at a randomly picked time | ||
schedule: `${randomInt(0, 59)} 6 * * *`, | ||
run: async function (app) { | ||
// need to iterate over invitations and send email to all over | ||
// 2 days old, but less than 3 days. | ||
const twoDays = new Date() | ||
twoDays.setDate(twoDays.getDate() - 2) | ||
const threeDays = new Date() | ||
threeDays.setDate(threeDays.getDate() - 3) | ||
const invites = await app.db.models.Invitation.findAll({ | ||
where: { | ||
createdAt: { | ||
[Op.between]: [threeDays, twoDays] | ||
}, | ||
reminderSentAt: { | ||
[Op.is]: null | ||
} | ||
}, | ||
include: [ | ||
{ model: app.db.models.User, as: 'invitor' }, | ||
{ model: app.db.models.User, as: 'invitee' }, | ||
{ model: app.db.models.Team, as: 'team' } | ||
] | ||
}) | ||
|
||
for (const invite of invites) { | ||
const expiryDate = invite.expiresAt.toDateString() | ||
let invitee = '' | ||
if (invite.invitee) { | ||
invitee = invite.invitee.name | ||
// Existing user | ||
await app.postoffice.send(invite.invitee, 'TeamInviteReminder', { | ||
teamName: invite.team.name, | ||
signupLink: `${app.config.base_url}/account/teams/invitations`, | ||
expiryDate | ||
}) | ||
} else if (invite.email) { | ||
invitee = invite.email | ||
// External user | ||
let signupLink = `${app.config.base_url}/account/create?email=${encodeURIComponent(invite.email)}` | ||
if (app.license.active()) { | ||
// Check if this is for an SSO-enabled domain with auto-create turned on | ||
const providerConfig = await app.db.models.SAMLProvider.forEmail(invite.email) | ||
if (providerConfig?.options?.provisionNewUsers) { | ||
signupLink = `${app.config.base_url}` | ||
} | ||
} | ||
|
||
await app.postoffice.send(invite, 'UnknownUserInvitationReminder', { | ||
invite, | ||
signupLink, | ||
expiryDate | ||
}) | ||
} | ||
invite.reminderSentAt = Date.now() | ||
await invite.save() | ||
|
||
// send reminder to Invitor | ||
await app.postoffice.send(invite.invitor, 'TeamInviterReminder', { | ||
teamName: invite.team.name, | ||
invitee, | ||
expiryDate | ||
}) | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
module.exports = { | ||
subject: 'Invitation to join team {{{teamName.text}}} on FlowFuse', | ||
text: | ||
`Hello! | ||
This is a reminder that you have an invite to join team {{{teamName.text}}} on the FlowFuse platform. | ||
This invitation will expire on {{{expiryDate}}}. | ||
{{{ signupLink }}} | ||
`, | ||
html: | ||
`<p>Hello!</p> | ||
<p>You've been invited to join team {{{teamName.html}}} on the FlowFuse platform.</p> | ||
<p>This invitation will expire on {{{expiryDate}}}.</p> | ||
<p><a href="{{{ signupLink }}}">Sign Up!</a></p> | ||
` | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,16 @@ | ||
module.exports = { | ||
subject: 'Invitation for {{{invitee.text}}} to {{{teamName.text}}} not accepted yet', | ||
text: | ||
` | ||
Hello, | ||
You invited {{{invitee.text}}} to join FlowFuse Team {{{teamName.text}}}, but they have not yet accepted. | ||
This invitation will expire on {{{expiryDate}}}. | ||
`, | ||
html: | ||
`<p>Hello</p> | ||
<p>You invited {{{invitee.html}}} to join FlowFuse Team {{{teamName.html}}}, but they have not yet accepted.</p> | ||
<p>This invitation will expire on {{{expiryDate}}}.</p> | ||
` | ||
} |
18 changes: 18 additions & 0 deletions
18
forge/postoffice/templates/UnknownUserInvitationReminder.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
module.exports = { | ||
subject: 'Invitation to collaborate on FlowFuse', | ||
text: | ||
`Hello! | ||
This is quick reminder that you've been invited to join the FlowFuse platform. Use the link below to sign-up and get started. | ||
This invitation will expire on {{{expiryDate}}}. | ||
{{{ signupLink }}} | ||
`, | ||
html: | ||
`<p>Hello!</p> | ||
<p>This is quick reminder that you've been invited to join the FlowFuse platform. Use the link below to sign-up and get started.</p> | ||
<p>This invitation will expire on {{{expiryDate}}}.</p> | ||
<p><a href="{{{ signupLink }}}">Sign Up!</a></p> | ||
` | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -346,4 +346,116 @@ describe('Team Invitations API', function () { | |
response.statusCode.should.equal(404) | ||
}) | ||
}) | ||
|
||
describe('Send invite reminders', async function () { | ||
before(function () { | ||
app.settings.set('team:user:invite:external', true) | ||
}) | ||
after(function () { | ||
app.settings.set('team:user:invite:external', false) | ||
}) | ||
it('Reminder should be sent after 2 days (internal)', async () => { | ||
const response = await app.inject({ | ||
method: 'POST', | ||
url: `/api/v1/teams/${TestObjects.BTeam.hashid}/invitations`, | ||
cookies: { sid: TestObjects.tokens.bob }, | ||
payload: { | ||
user: 'chris' | ||
} | ||
}) | ||
const result = response.json() | ||
result.should.have.property('status', 'okay') | ||
const invites = await app.db.models.Invitation.findAll({ | ||
where: { | ||
inviteeId: TestObjects.chris.id | ||
} | ||
}) | ||
const origTime = invites[0].createdAt | ||
origTime.setDate(origTime.getDate() - 2) | ||
origTime.setHours(origTime.getHours() - 2) | ||
invites[0].createdAt = origTime | ||
invites[0].changed('createdAt', true) | ||
await invites[0].save() | ||
|
||
const houseKeepingJob = require('../../../../../forge/housekeeper/tasks/inviteReminder') | ||
await houseKeepingJob.run(app) | ||
app.config.email.transport.getMessageQueue().should.have.lengthOf(3) | ||
app.config.email.transport.getMessageQueue()[1].to.should.equal(TestObjects.chris.email) | ||
app.config.email.transport.getMessageQueue()[1].subject.should.equal('Invitation to join team BTeam on FlowFuse') | ||
app.config.email.transport.getMessageQueue()[2].to.should.equal(TestObjects.bob.email) | ||
app.config.email.transport.getMessageQueue()[2].subject.should.equal('Invitation for Chris Kenobi to BTeam not accepted yet') | ||
}) | ||
it('Reminder should be sent after 2 days (external)', async () => { | ||
const response = await app.inject({ | ||
method: 'POST', | ||
url: `/api/v1/teams/${TestObjects.BTeam.hashid}/invitations`, | ||
cookies: { sid: TestObjects.tokens.bob }, | ||
payload: { | ||
user: '[email protected]' | ||
} | ||
}) | ||
const result = response.json() | ||
result.should.have.property('status', 'okay') | ||
const invites = await app.db.models.Invitation.findAll({ | ||
where: { | ||
email: '[email protected]' | ||
} | ||
}) | ||
const origTime = invites[0].createdAt | ||
origTime.setDate(origTime.getDate() - 2) | ||
origTime.setHours(origTime.getHours() - 2) | ||
invites[0].createdAt = origTime | ||
invites[0].changed('createdAt', true) | ||
await invites[0].save() | ||
|
||
const houseKeepingJob = require('../../../../../forge/housekeeper/tasks/inviteReminder') | ||
await houseKeepingJob.run(app) | ||
app.config.email.transport.getMessageQueue().should.have.lengthOf(3) | ||
app.config.email.transport.getMessageQueue()[1].to.should.equal('[email protected]') | ||
app.config.email.transport.getMessageQueue()[1].subject.should.equal('Invitation to collaborate on FlowFuse') | ||
app.config.email.transport.getMessageQueue()[2].to.should.equal(TestObjects.bob.email) | ||
app.config.email.transport.getMessageQueue()[2].subject.should.equal('Invitation for evans@example com to BTeam not accepted yet') | ||
}) | ||
}) | ||
|
||
describe('Delete expired invites', async function () { | ||
before(function () { | ||
app.settings.set('team:user:invite:external', true) | ||
}) | ||
after(function () { | ||
app.settings.set('team:user:invite:external', false) | ||
}) | ||
it('Delete invites after 7 days', async () => { | ||
const response = await app.inject({ | ||
method: 'POST', | ||
url: `/api/v1/teams/${TestObjects.BTeam.hashid}/invitations`, | ||
cookies: { sid: TestObjects.tokens.bob }, | ||
payload: { | ||
user: '[email protected]' | ||
} | ||
}) | ||
const result = response.json() | ||
result.should.have.property('status', 'okay') | ||
const invites = await app.db.models.Invitation.findAll({ | ||
where: { | ||
email: '[email protected]' | ||
} | ||
}) | ||
const origTime = invites[0].expiresAt | ||
origTime.setDate(origTime.getDate() - 8) | ||
invites[0].expiresAt = origTime | ||
invites[0].changed('expiresAt', true) | ||
await invites[0].save() | ||
|
||
const houseKeepingJob = require('../../../../../forge/housekeeper/tasks/expireInvites') | ||
await houseKeepingJob.run(app) | ||
|
||
const noInvites = await app.db.models.Invitation.findAll({ | ||
where: { | ||
email: '[email protected]' | ||
} | ||
}) | ||
noInvites.should.have.lengthOf(0) | ||
}) | ||
}) | ||
}) |