mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-17 14:47:53 +01:00
* refactor: Move translate test utility to helpers directory * Add kind property to webhooks * feat: Add options to create webhook route * refactor: Move webhook ops into single file * refactor: Create webhook objects for specific webhook behavior * chore(tests): Add default sleep helper value of 1 second * feat(api): Add method for groups to send out webhook * feat(api): Add taskCreated webhook task creation * feat(api): Send chat webhooks after a chat is sent * refactor: Move webhook routes to own controller * lint: Correct linting errors * fix(api): Correct taskCreated webhook method * fix(api): Fix webhook logging to only log when there is an error * fix: Update groupChatRecieved webhook creation * chore: Add integration tests for webhooks * fix: Set webhook creation response to 201 * fix: Correct how task scored webhook data is sent * Revert group chat recieved webhook to only support one group * Remove quest activity option for webhooks * feat: Send webhook for each task created * feat: Allow webhooks without a type to default to taskScored * feat: Add logic for adding ids to webhook * feat: optimize webhook url check by shortcircuiting if no url is passed * refactor: Use full name for webhook variable * feat: Add missing params to client webhook * lint: Add missing semicolon * chore(tests): Fix inccorect webhook tests * chore: Add migration to update task scored webhooks * feat: Allow default value of webhook add route to be enabled * chore: Update webhook documentation * chore: Remove special handling for v2 * refactor: adjust addComputedStatsToJSONObject to work for webhooks * refactor: combine taskScored and taskActivity webhooks * feat(api): Add task activity to task update and delete routes * chore: Change references to taskScored to taskActivity * fix: Correct stats object being passed in for transform * chore: Remove extra line break * fix: Pass in the language to use for the translations * refactor(api): Move webhooks from user.preferences.webhooks to user.webhooks * chore: Update migration to set webhook array * lint: Correct brace spacing * chore: convert webhook lib to use user.webhooks * refactor(api): Consolidate filters * chore: clarify migration instructions * fix(test): Correct user creation in user anonymized tests * chore: add test that webhooks cannot be updated via PUT /user * refactor: Simplify default webhook id value * refactor(client): Push newly created webhook instead of doing a sync * chore(test): Add test file for webhook model * refactor: Remove webhook validation * refactor: Remove need for watch on webhooks * refactor(client): Update webhooks object without syncing * chore: update webhook documentation * Fix migrations issues * chore: remove v2 test helper * fix(api): Provide webhook type in task scored webhook * fix(client): Fix webhook deletion appearing to delete all webhooks * feat(api): add optional label field for webhooks * feat: provide empty string as default for webhook label * chore: Update webhook migration * chore: update webhook migration name
512 lines
14 KiB
JavaScript
512 lines
14 KiB
JavaScript
import {
|
|
generateUser,
|
|
generateGroup,
|
|
sleep,
|
|
generateChallenge,
|
|
server,
|
|
} from '../../../../helpers/api-integration/v3';
|
|
import { v4 as generateUUID } from 'uuid';
|
|
|
|
describe('PUT /tasks/:id', () => {
|
|
let user;
|
|
|
|
beforeEach(async () => {
|
|
user = await generateUser();
|
|
});
|
|
|
|
context('validates params', () => {
|
|
let task;
|
|
|
|
beforeEach(async () => {
|
|
task = await user.post('/tasks/user', {
|
|
text: 'test habit',
|
|
type: 'habit',
|
|
});
|
|
});
|
|
|
|
it(`ignores setting _id, type, userId, history, createdAt,
|
|
updatedAt, challenge, completed, streak,
|
|
dateCompleted fields`, async () => {
|
|
let savedTask = await user.put(`/tasks/${task._id}`, {
|
|
_id: 123,
|
|
type: 'daily',
|
|
userId: 123,
|
|
history: [123],
|
|
createdAt: 'yesterday',
|
|
updatedAt: 'tomorrow',
|
|
challenge: 'no',
|
|
completed: true,
|
|
streak: 25,
|
|
dateCompleted: 'never',
|
|
});
|
|
|
|
expect(savedTask._id).to.equal(task._id);
|
|
expect(savedTask.type).to.equal(task.type);
|
|
expect(savedTask.userId).to.equal(task.userId);
|
|
expect(savedTask.history).to.eql(task.history);
|
|
expect(savedTask.createdAt).to.equal(task.createdAt);
|
|
expect(savedTask.updatedAt).to.be.greaterThan(task.updatedAt);
|
|
expect(savedTask.challenge).to.eql(task.challenge);
|
|
expect(savedTask.completed).to.eql(task.completed);
|
|
expect(savedTask.streak).to.equal(savedTask.streak); // it's an habit, dailies can change it
|
|
expect(savedTask.dateCompleted).to.equal(task.dateCompleted);
|
|
});
|
|
|
|
it('ignores invalid fields', async () => {
|
|
let savedTask = await user.put(`/tasks/${task._id}`, {
|
|
notValid: true,
|
|
});
|
|
|
|
expect(savedTask.notValid).to.be.undefined;
|
|
});
|
|
|
|
it(`only allows setting streak, alias, reminders, checklist, notes, attribute, tags
|
|
fields for challenge tasks owned by a user`, async () => {
|
|
let guild = await generateGroup(user);
|
|
let challenge = await generateChallenge(user, guild);
|
|
|
|
let challengeTask = await user.post(`/tasks/challenge/${challenge._id}`, {
|
|
type: 'daily',
|
|
text: 'Daily in challenge',
|
|
reminders: [
|
|
{time: new Date(), startDate: new Date()},
|
|
],
|
|
checklist: [
|
|
{text: 123, completed: false},
|
|
],
|
|
});
|
|
await sleep(2);
|
|
|
|
await user.sync();
|
|
|
|
// Pick challenge task
|
|
let challengeUserTaskId = user.tasksOrder.dailys[user.tasksOrder.dailys.length - 1];
|
|
|
|
let challengeUserTask = await user.get(`/tasks/${challengeUserTaskId}`);
|
|
|
|
let savedChallengeUserTask = await user.put(`/tasks/${challengeUserTaskId}`, {
|
|
_id: 123,
|
|
type: 'daily',
|
|
userId: 123,
|
|
alias: 'a-short-task-name',
|
|
history: [123],
|
|
createdAt: 'yesterday',
|
|
updatedAt: 'tomorrow',
|
|
challenge: 'no',
|
|
completed: true,
|
|
streak: 25,
|
|
priority: 1.5,
|
|
repeat: {
|
|
m: false,
|
|
},
|
|
everyX: 15,
|
|
frequency: 'weekly',
|
|
text: 'new text',
|
|
dateCompleted: 'never',
|
|
reminders: [
|
|
{time: new Date(), startDate: new Date()},
|
|
{time: new Date(), startDate: new Date()},
|
|
],
|
|
checklist: [
|
|
{text: 123, completed: false},
|
|
{text: 456, completed: true},
|
|
],
|
|
notes: 'new notes',
|
|
attribute: 'per',
|
|
tags: [challengeUserTaskId],
|
|
});
|
|
|
|
// original task is not touched
|
|
let updatedChallengeTask = await user.get(`/tasks/${challengeTask._id}`);
|
|
expect(updatedChallengeTask).to.eql(challengeTask);
|
|
|
|
// ignored
|
|
expect(savedChallengeUserTask._id).to.equal(challengeUserTask._id);
|
|
expect(savedChallengeUserTask.type).to.equal(challengeUserTask.type);
|
|
expect(savedChallengeUserTask.repeat.m).to.equal(true);
|
|
expect(savedChallengeUserTask.priority).to.equal(challengeUserTask.priority);
|
|
expect(savedChallengeUserTask.frequency).to.equal(challengeUserTask.frequency);
|
|
expect(savedChallengeUserTask.userId).to.equal(challengeUserTask.userId);
|
|
expect(savedChallengeUserTask.text).to.equal(challengeUserTask.text);
|
|
expect(savedChallengeUserTask.history).to.eql(challengeUserTask.history);
|
|
expect(savedChallengeUserTask.createdAt).to.equal(challengeUserTask.createdAt);
|
|
expect(savedChallengeUserTask.updatedAt).to.be.greaterThan(challengeUserTask.updatedAt);
|
|
expect(savedChallengeUserTask.challenge).to.eql(challengeUserTask.challenge);
|
|
expect(savedChallengeUserTask.completed).to.equal(challengeUserTask.completed);
|
|
expect(savedChallengeUserTask.dateCompleted).to.equal(challengeUserTask.dateCompleted);
|
|
expect(savedChallengeUserTask.priority).to.equal(challengeUserTask.priority);
|
|
|
|
// changed
|
|
expect(savedChallengeUserTask.notes).to.equal('new notes');
|
|
expect(savedChallengeUserTask.attribute).to.equal('per');
|
|
expect(savedChallengeUserTask.tags).to.eql([challengeUserTaskId]);
|
|
expect(savedChallengeUserTask.streak).to.equal(25);
|
|
expect(savedChallengeUserTask.reminders.length).to.equal(2);
|
|
expect(savedChallengeUserTask.checklist.length).to.equal(2);
|
|
});
|
|
});
|
|
|
|
context('sending task activity webhooks', () => {
|
|
before(async () => {
|
|
await server.start();
|
|
});
|
|
|
|
after(async () => {
|
|
await server.close();
|
|
});
|
|
|
|
it('sends task activity webhooks if task is user owned', async () => {
|
|
let uuid = generateUUID();
|
|
|
|
await user.post('/user/webhook', {
|
|
url: `http://localhost:${server.port}/webhooks/${uuid}`,
|
|
type: 'taskActivity',
|
|
enabled: true,
|
|
options: {
|
|
created: false,
|
|
updated: true,
|
|
},
|
|
});
|
|
|
|
let task = await user.post('/tasks/user', {
|
|
text: 'test habit',
|
|
type: 'habit',
|
|
});
|
|
|
|
let updatedTask = await user.put(`/tasks/${task.id}`, {
|
|
text: 'updated text',
|
|
});
|
|
|
|
await sleep();
|
|
|
|
let body = server.getWebhookData(uuid);
|
|
|
|
expect(body.type).to.eql('updated');
|
|
expect(body.task).to.eql(updatedTask);
|
|
});
|
|
|
|
it('does not send task activity webhooks if task is not user owned', async () => {
|
|
let uuid = generateUUID();
|
|
|
|
await user.update({
|
|
balance: 10,
|
|
});
|
|
let guild = await generateGroup(user);
|
|
let challenge = await generateChallenge(user, guild);
|
|
|
|
await user.post('/user/webhook', {
|
|
url: `http://localhost:${server.port}/webhooks/${uuid}`,
|
|
type: 'taskActivity',
|
|
enabled: true,
|
|
options: {
|
|
created: false,
|
|
updated: true,
|
|
},
|
|
});
|
|
|
|
let task = await user.post(`/tasks/challenge/${challenge._id}`, {
|
|
text: 'test habit',
|
|
type: 'habit',
|
|
});
|
|
|
|
await user.put(`/tasks/${task.id}`, {
|
|
text: 'updated text',
|
|
});
|
|
|
|
await sleep();
|
|
|
|
let body = server.getWebhookData(uuid);
|
|
|
|
expect(body).to.not.exist;
|
|
});
|
|
});
|
|
|
|
context('all types', () => {
|
|
let daily;
|
|
|
|
beforeEach(async () => {
|
|
daily = await user.post('/tasks/user', {
|
|
text: 'test daily',
|
|
type: 'daily',
|
|
notes: 1976,
|
|
});
|
|
});
|
|
|
|
it('can update reminders (replace them)', async () => {
|
|
await user.put(`/tasks/${daily._id}`, {
|
|
reminders: [
|
|
{time: new Date(), startDate: new Date()},
|
|
],
|
|
});
|
|
|
|
let id1 = generateUUID();
|
|
let id2 = generateUUID();
|
|
|
|
let savedDaily = await user.put(`/tasks/${daily._id}`, {
|
|
reminders: [
|
|
{id: id1, time: new Date(), startDate: new Date()},
|
|
{id: id2, time: new Date(), startDate: new Date()},
|
|
],
|
|
});
|
|
|
|
expect(savedDaily.reminders.length).to.equal(2);
|
|
expect(savedDaily.reminders[0].id).to.equal(id1);
|
|
expect(savedDaily.reminders[1].id).to.equal(id2);
|
|
});
|
|
|
|
it('can set a alias if no other task has that alias', async () => {
|
|
let savedDaily = await user.put(`/tasks/${daily._id}`, {
|
|
alias: 'alias',
|
|
});
|
|
|
|
expect(savedDaily.alias).to.eql('alias');
|
|
});
|
|
|
|
it('does not set alias to a alias that is already in use', async () => {
|
|
await user.post('/tasks/user', {
|
|
type: 'todo',
|
|
text: 'a todo',
|
|
alias: 'some-alias',
|
|
});
|
|
|
|
await expect(user.put(`/tasks/${daily._id}`, {
|
|
alias: 'some-alias',
|
|
})).to.eventually.be.rejected.and.eql({
|
|
code: 400,
|
|
error: 'BadRequest',
|
|
message: 'daily validation failed',
|
|
});
|
|
});
|
|
|
|
it('can use alias to update a task', async () => {
|
|
daily = await user.put(`/tasks/${daily._id}`, {
|
|
alias: 'alias',
|
|
});
|
|
|
|
await user.put(`/tasks/${daily.alias}`, {
|
|
text: 'saved',
|
|
});
|
|
|
|
let fetchedDaily = await user.get(`/tasks/${daily._id}`);
|
|
|
|
expect(fetchedDaily.text).to.eql('saved');
|
|
});
|
|
});
|
|
|
|
context('habits', () => {
|
|
let habit;
|
|
|
|
beforeEach(async () => {
|
|
habit = await user.post('/tasks/user', {
|
|
text: 'test habit',
|
|
type: 'habit',
|
|
notes: 1976,
|
|
});
|
|
});
|
|
|
|
it('updates a habit', async () => {
|
|
let savedHabit = await user.put(`/tasks/${habit._id}`, {
|
|
text: 'some new text',
|
|
up: false,
|
|
down: false,
|
|
notes: 'some new notes',
|
|
});
|
|
|
|
expect(savedHabit.text).to.eql('some new text');
|
|
expect(savedHabit.notes).to.eql('some new notes');
|
|
expect(savedHabit.up).to.eql(false);
|
|
expect(savedHabit.down).to.eql(false);
|
|
});
|
|
});
|
|
|
|
context('todos', () => {
|
|
let todo;
|
|
|
|
beforeEach(async () => {
|
|
todo = await user.post('/tasks/user', {
|
|
text: 'test todo',
|
|
type: 'todo',
|
|
notes: 1976,
|
|
});
|
|
});
|
|
|
|
it('updates a todo', async () => {
|
|
let savedTodo = await user.put(`/tasks/${todo._id}`, {
|
|
text: 'some new text',
|
|
notes: 'some new notes',
|
|
});
|
|
|
|
expect(savedTodo.text).to.eql('some new text');
|
|
expect(savedTodo.notes).to.eql('some new notes');
|
|
});
|
|
|
|
it('can update checklists (replace it)', async () => {
|
|
await user.put(`/tasks/${todo._id}`, {
|
|
checklist: [
|
|
{text: 123, completed: false},
|
|
{text: 456, completed: true},
|
|
],
|
|
});
|
|
|
|
let savedTodo = await user.put(`/tasks/${todo._id}`, {
|
|
checklist: [
|
|
{text: 789, completed: false},
|
|
],
|
|
});
|
|
|
|
expect(savedTodo.checklist.length).to.equal(1);
|
|
expect(savedTodo.checklist[0].text).to.equal('789');
|
|
expect(savedTodo.checklist[0].completed).to.equal(false);
|
|
});
|
|
|
|
it('can update tags (replace them)', async () => {
|
|
let finalUUID = generateUUID();
|
|
await user.put(`/tasks/${todo._id}`, {
|
|
tags: [generateUUID(), generateUUID()],
|
|
});
|
|
|
|
let savedTodo = await user.put(`/tasks/${todo._id}`, {
|
|
tags: [finalUUID],
|
|
});
|
|
|
|
expect(savedTodo.tags.length).to.equal(1);
|
|
expect(savedTodo.tags[0]).to.equal(finalUUID);
|
|
});
|
|
});
|
|
|
|
context('dailys', () => {
|
|
let daily;
|
|
|
|
beforeEach(async () => {
|
|
daily = await user.post('/tasks/user', {
|
|
text: 'test daily',
|
|
type: 'daily',
|
|
notes: 1976,
|
|
});
|
|
});
|
|
|
|
it('updates a daily', async () => {
|
|
let savedDaily = await user.put(`/tasks/${daily._id}`, {
|
|
text: 'some new text',
|
|
notes: 'some new notes',
|
|
frequency: 'daily',
|
|
everyX: 5,
|
|
});
|
|
|
|
expect(savedDaily.text).to.eql('some new text');
|
|
expect(savedDaily.notes).to.eql('some new notes');
|
|
expect(savedDaily.frequency).to.eql('daily');
|
|
expect(savedDaily.everyX).to.eql(5);
|
|
});
|
|
|
|
it('can update checklists (replace it)', async () => {
|
|
await user.put(`/tasks/${daily._id}`, {
|
|
checklist: [
|
|
{text: 123, completed: false},
|
|
{text: 456, completed: true},
|
|
],
|
|
});
|
|
|
|
let savedDaily = await user.put(`/tasks/${daily._id}`, {
|
|
checklist: [
|
|
{text: 789, completed: false},
|
|
],
|
|
});
|
|
|
|
expect(savedDaily.checklist.length).to.equal(1);
|
|
expect(savedDaily.checklist[0].text).to.equal('789');
|
|
expect(savedDaily.checklist[0].completed).to.equal(false);
|
|
});
|
|
|
|
it('can update tags (replace them)', async () => {
|
|
let finalUUID = generateUUID();
|
|
await user.put(`/tasks/${daily._id}`, {
|
|
tags: [generateUUID(), generateUUID()],
|
|
});
|
|
|
|
let savedDaily = await user.put(`/tasks/${daily._id}`, {
|
|
tags: [finalUUID],
|
|
});
|
|
|
|
expect(savedDaily.tags.length).to.equal(1);
|
|
expect(savedDaily.tags[0]).to.equal(finalUUID);
|
|
});
|
|
|
|
it('updates repeat, even if frequency is set to daily', async () => {
|
|
await user.put(`/tasks/${daily._id}`, {
|
|
frequency: 'daily',
|
|
});
|
|
|
|
let savedDaily = await user.put(`/tasks/${daily._id}`, {
|
|
repeat: {
|
|
m: false,
|
|
su: false,
|
|
},
|
|
});
|
|
|
|
expect(savedDaily.repeat).to.eql({
|
|
m: false,
|
|
t: true,
|
|
w: true,
|
|
th: true,
|
|
f: true,
|
|
s: true,
|
|
su: false,
|
|
});
|
|
});
|
|
|
|
it('updates everyX, even if frequency is set to weekly', async () => {
|
|
await user.put(`/tasks/${daily._id}`, {
|
|
frequency: 'weekly',
|
|
});
|
|
|
|
let savedDaily = await user.put(`/tasks/${daily._id}`, {
|
|
everyX: 5,
|
|
});
|
|
|
|
expect(savedDaily.everyX).to.eql(5);
|
|
});
|
|
|
|
it('defaults startDate to today if none date object is passed in', async () => {
|
|
let savedDaily = await user.put(`/tasks/${daily._id}`, {
|
|
frequency: 'weekly',
|
|
});
|
|
|
|
expect((new Date(savedDaily.startDate)).getDay()).to.eql((new Date()).getDay());
|
|
});
|
|
});
|
|
|
|
context('rewards', () => {
|
|
let reward;
|
|
|
|
beforeEach(async () => {
|
|
reward = await user.post('/tasks/user', {
|
|
text: 'test reward',
|
|
type: 'reward',
|
|
notes: 1976,
|
|
value: 10,
|
|
});
|
|
});
|
|
|
|
it('updates a reward', async () => {
|
|
let savedReward = await user.put(`/tasks/${reward._id}`, {
|
|
text: 'some new text',
|
|
notes: 'some new notes',
|
|
value: 10,
|
|
});
|
|
|
|
expect(savedReward.text).to.eql('some new text');
|
|
expect(savedReward.notes).to.eql('some new notes');
|
|
expect(savedReward.value).to.eql(10);
|
|
});
|
|
|
|
it('requires value to be coerced into a number', async () => {
|
|
let savedReward = await user.put(`/tasks/${reward._id}`, {
|
|
value: '100',
|
|
});
|
|
|
|
expect(savedReward.value).to.eql(100);
|
|
});
|
|
});
|
|
});
|