mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-17 14:47:53 +01:00
Merge pull request #7043 from TheHollidayInn/api-v3-ops-reset
Ported reset. Added unit tests. Added reset route. Added integration tests
This commit is contained in:
@@ -168,5 +168,6 @@
|
|||||||
"resetComplete": "Reset has completed",
|
"resetComplete": "Reset has completed",
|
||||||
"regIdRequired": "RegId is required",
|
"regIdRequired": "RegId is required",
|
||||||
"pushDeviceAdded": "Push device added successfully",
|
"pushDeviceAdded": "Push device added successfully",
|
||||||
"pushDeviceAlreadyAdded": "The user already has the push device"
|
"pushDeviceAlreadyAdded": "The user already has the push device",
|
||||||
|
"resetComplete": "Reset has completed"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -120,6 +120,7 @@ import clearPMs from './ops/clearPMs';
|
|||||||
import deletePM from './ops/deletePM';
|
import deletePM from './ops/deletePM';
|
||||||
import reroll from './ops/reroll';
|
import reroll from './ops/reroll';
|
||||||
import addPushDevice from './ops/addPushDevice';
|
import addPushDevice from './ops/addPushDevice';
|
||||||
|
import reset from './ops/reset';
|
||||||
|
|
||||||
api.ops = {
|
api.ops = {
|
||||||
scoreTask,
|
scoreTask,
|
||||||
@@ -156,6 +157,7 @@ api.ops = {
|
|||||||
deletePM,
|
deletePM,
|
||||||
reroll,
|
reroll,
|
||||||
addPushDevice,
|
addPushDevice,
|
||||||
|
reset,
|
||||||
};
|
};
|
||||||
|
|
||||||
import handleTwoHanded from './fns/handleTwoHanded';
|
import handleTwoHanded from './fns/handleTwoHanded';
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import { MAX_LEVEL } from '../constants';
|
|||||||
import {
|
import {
|
||||||
NotAuthorized,
|
NotAuthorized,
|
||||||
} from '../libs/errors';
|
} from '../libs/errors';
|
||||||
import resetGear from './resetGear';
|
import resetGear from '../fns/resetGear';
|
||||||
import equip from './equip';
|
import equip from './equip';
|
||||||
|
|
||||||
const USERSTATSLIST = ['per', 'int', 'con', 'str', 'points', 'gp', 'exp', 'mp'];
|
const USERSTATSLIST = ['per', 'int', 'con', 'str', 'points', 'gp', 'exp', 'mp'];
|
||||||
|
|||||||
@@ -1,35 +1,28 @@
|
|||||||
import _ from 'lodash';
|
import resetGear from '../fns/resetGear';
|
||||||
|
import i18n from '../i18n';
|
||||||
|
|
||||||
module.exports = function(user, req, cb) {
|
module.exports = function reset (user, tasks = []) {
|
||||||
var gear;
|
|
||||||
user.habits = [];
|
|
||||||
user.dailys = [];
|
|
||||||
user.todos = [];
|
|
||||||
user.rewards = [];
|
|
||||||
user.stats.hp = 50;
|
user.stats.hp = 50;
|
||||||
user.stats.lvl = 1;
|
user.stats.lvl = 1;
|
||||||
user.stats.gp = 0;
|
user.stats.gp = 0;
|
||||||
user.stats.exp = 0;
|
user.stats.exp = 0;
|
||||||
gear = user.items.gear;
|
|
||||||
_.each(['equipped', 'costume'], function(type) {
|
let tasksToRemove = [];
|
||||||
gear[type].armor = 'armor_base_0';
|
tasks.forEach(task => {
|
||||||
gear[type].weapon = 'weapon_base_0';
|
if (!task.challenge || !task.challenge.id || task.challenge.broken) {
|
||||||
gear[type].head = 'head_base_0';
|
tasksToRemove.push(task._id);
|
||||||
return gear[type].shield = 'shield_base_0';
|
let i = user.tasksOrder[`${task.type}s`].indexOf(task._id);
|
||||||
});
|
if (i !== -1) user.tasksOrder[`${task.type}s`].splice(i, 1);
|
||||||
if (typeof gear.owned === 'undefined') {
|
tasksToRemove.push(task._id);
|
||||||
gear.owned = {};
|
|
||||||
}
|
|
||||||
_.each(gear.owned, function(v, k) {
|
|
||||||
if (gear.owned[k]) {
|
|
||||||
gear.owned[k] = false;
|
|
||||||
}
|
}
|
||||||
return true;
|
|
||||||
});
|
});
|
||||||
gear.owned.weapon_warrior_0 = true;
|
|
||||||
if (typeof user.markModified === "function") {
|
resetGear(user);
|
||||||
user.markModified('items.gear.owned');
|
|
||||||
}
|
let response = {
|
||||||
user.preferences.costume = false;
|
data: {user, tasksToRemove},
|
||||||
return typeof cb === "function" ? cb(null, user) : void 0;
|
message: i18n.t('resetComplete'),
|
||||||
|
};
|
||||||
|
|
||||||
|
return response;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ const COMMON_FILES = [
|
|||||||
'./common/script/**/*.js',
|
'./common/script/**/*.js',
|
||||||
// @TODO remove these negations as the files are converted over.
|
// @TODO remove these negations as the files are converted over.
|
||||||
'!./common/script/content/index.js',
|
'!./common/script/content/index.js',
|
||||||
'!./common/script/ops/reset.js',
|
|
||||||
'!./common/script/fns/randomDrop.js',
|
'!./common/script/fns/randomDrop.js',
|
||||||
'!./common/script/libs/countExists.js',
|
'!./common/script/libs/countExists.js',
|
||||||
'!./common/script/libs/encodeiCalLink.js',
|
'!./common/script/libs/encodeiCalLink.js',
|
||||||
|
|||||||
104
test/api/v3/integration/user/POST-user_reset.test.js
Normal file
104
test/api/v3/integration/user/POST-user_reset.test.js
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
import {
|
||||||
|
generateUser,
|
||||||
|
generateGroup,
|
||||||
|
generateChallenge,
|
||||||
|
translate as t,
|
||||||
|
} from '../../../../helpers/api-integration/v3';
|
||||||
|
|
||||||
|
describe('POST /user/reset', () => {
|
||||||
|
let user;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
user = await generateUser();
|
||||||
|
});
|
||||||
|
|
||||||
|
// More tests in common code unit tests
|
||||||
|
|
||||||
|
it('resets user\'s habits', async () => {
|
||||||
|
let task = await user.post('/tasks/user', {
|
||||||
|
text: 'test habit',
|
||||||
|
type: 'habit',
|
||||||
|
});
|
||||||
|
|
||||||
|
await user.post('/user/reset');
|
||||||
|
await user.sync();
|
||||||
|
|
||||||
|
await expect(user.get(`/tasks/${task._id}`)).to.eventually.be.rejected.and.eql({
|
||||||
|
code: 404,
|
||||||
|
error: 'NotFound',
|
||||||
|
message: t('taskNotFound'),
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(user.tasksOrder.habits).to.be.empty;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resets user\'s dailys', async () => {
|
||||||
|
let task = await user.post('/tasks/user', {
|
||||||
|
text: 'test daily',
|
||||||
|
type: 'daily',
|
||||||
|
});
|
||||||
|
|
||||||
|
await user.post('/user/reset');
|
||||||
|
await user.sync();
|
||||||
|
|
||||||
|
await expect(user.get(`/tasks/${task._id}`)).to.eventually.be.rejected.and.eql({
|
||||||
|
code: 404,
|
||||||
|
error: 'NotFound',
|
||||||
|
message: t('taskNotFound'),
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(user.tasksOrder.dailys).to.be.empty;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resets user\'s todos', async () => {
|
||||||
|
let task = await user.post('/tasks/user', {
|
||||||
|
text: 'test todo',
|
||||||
|
type: 'todo',
|
||||||
|
});
|
||||||
|
|
||||||
|
await user.post('/user/reset');
|
||||||
|
await user.sync();
|
||||||
|
|
||||||
|
await expect(user.get(`/tasks/${task._id}`)).to.eventually.be.rejected.and.eql({
|
||||||
|
code: 404,
|
||||||
|
error: 'NotFound',
|
||||||
|
message: t('taskNotFound'),
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(user.tasksOrder.todos).to.be.empty;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resets user\'s rewards', async () => {
|
||||||
|
let task = await user.post('/tasks/user', {
|
||||||
|
text: 'test reward',
|
||||||
|
type: 'reward',
|
||||||
|
});
|
||||||
|
|
||||||
|
await user.post('/user/reset');
|
||||||
|
await user.sync();
|
||||||
|
|
||||||
|
await expect(user.get(`/tasks/${task._id}`)).to.eventually.be.rejected.and.eql({
|
||||||
|
code: 404,
|
||||||
|
error: 'NotFound',
|
||||||
|
message: t('taskNotFound'),
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(user.tasksOrder.rewards).to.be.empty;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not delete challenge tasks', async () => {
|
||||||
|
let guild = await generateGroup(user);
|
||||||
|
let challenge = await generateChallenge(user, guild);
|
||||||
|
let task = await user.post(`/tasks/challenge/${challenge._id}`, {
|
||||||
|
text: 'test challenge habit',
|
||||||
|
type: 'habit',
|
||||||
|
});
|
||||||
|
|
||||||
|
await user.post('/user/reset');
|
||||||
|
await user.sync();
|
||||||
|
|
||||||
|
let userChallengeTask = await user.get(`/tasks/${task._id}`);
|
||||||
|
|
||||||
|
expect(userChallengeTask).to.eql(task);
|
||||||
|
});
|
||||||
|
});
|
||||||
79
test/common/ops/reset.js
Normal file
79
test/common/ops/reset.js
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
import reset from '../../../common/script/ops/reset';
|
||||||
|
import i18n from '../../../common/script/i18n';
|
||||||
|
import {
|
||||||
|
generateUser,
|
||||||
|
generateDaily,
|
||||||
|
generateHabit,
|
||||||
|
generateReward,
|
||||||
|
generateTodo,
|
||||||
|
} from '../../helpers/common.helper';
|
||||||
|
|
||||||
|
describe('shared.ops.reset', () => {
|
||||||
|
let user;
|
||||||
|
let tasksToRemove;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
user = generateUser();
|
||||||
|
user.balance = 2;
|
||||||
|
|
||||||
|
let habit = generateHabit();
|
||||||
|
let todo = generateTodo();
|
||||||
|
let daily = generateDaily();
|
||||||
|
let reward = generateReward();
|
||||||
|
|
||||||
|
user.tasksOrder.habits = [habit._id];
|
||||||
|
user.tasksOrder.todos = [todo._id];
|
||||||
|
user.tasksOrder.dailys = [daily._id];
|
||||||
|
user.tasksOrder.rewards = [reward._id];
|
||||||
|
|
||||||
|
tasksToRemove = [habit, todo, daily, reward];
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
it('resets a user', () => {
|
||||||
|
let response = reset(user);
|
||||||
|
|
||||||
|
expect(response.message).to.equal(i18n.t('resetComplete'));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resets user\'s health', () => {
|
||||||
|
user.stats.hp = 40;
|
||||||
|
|
||||||
|
reset(user);
|
||||||
|
|
||||||
|
expect(user.stats.hp).to.equal(50);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resets user\'s level', () => {
|
||||||
|
user.stats.lvl = 2;
|
||||||
|
|
||||||
|
reset(user);
|
||||||
|
|
||||||
|
expect(user.stats.lvl).to.equal(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resets user\'s gold', () => {
|
||||||
|
user.stats.gp = 20;
|
||||||
|
|
||||||
|
reset(user);
|
||||||
|
|
||||||
|
expect(user.stats.gp).to.equal(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resets user\'s exp', () => {
|
||||||
|
user.stats.exp = 20;
|
||||||
|
|
||||||
|
reset(user);
|
||||||
|
|
||||||
|
expect(user.stats.exp).to.equal(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('resets user\'s tasksOrder', () => {
|
||||||
|
reset(user, tasksToRemove);
|
||||||
|
|
||||||
|
expect(user.tasksOrder.habits).to.be.empty;
|
||||||
|
expect(user.tasksOrder.todos).to.be.empty;
|
||||||
|
expect(user.tasksOrder.dailys).to.be.empty;
|
||||||
|
expect(user.tasksOrder.rewards).to.be.empty;
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -1083,4 +1083,29 @@ api.userAddPushDevice = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @api {post} /user/reset Resets a user.
|
||||||
|
* @apiVersion 3.0.0
|
||||||
|
* @apiName UserReset
|
||||||
|
* @apiGroup User
|
||||||
|
*
|
||||||
|
* @apiSuccess {Object} data `user`
|
||||||
|
*/
|
||||||
|
api.userReset = {
|
||||||
|
method: 'POST',
|
||||||
|
middlewares: [authWithHeaders(), cron],
|
||||||
|
url: '/user/reset',
|
||||||
|
async handler (req, res) {
|
||||||
|
let user = res.locals.user;
|
||||||
|
|
||||||
|
let tasks = await Tasks.Task.find({userId: user._id}).select('_id type challenge').exec();
|
||||||
|
|
||||||
|
let resetResponse = common.ops.reset(user, tasks);
|
||||||
|
|
||||||
|
await Q.all([Tasks.Task.remove({_id: {$in: resetResponse.data.tasksToRemove}, userId: user._id}), user.save()]);
|
||||||
|
|
||||||
|
res.respond(200, resetResponse);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = api;
|
module.exports = api;
|
||||||
|
|||||||
@@ -808,7 +808,6 @@ schema.methods.getTransformedData = function getTransformedData (cb) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// END of API v2 methods
|
// END of API v2 methods
|
||||||
|
|
||||||
export let model = mongoose.model('User', schema);
|
export let model = mongoose.model('User', schema);
|
||||||
|
|
||||||
// Initially export an empty object so external requires will get
|
// Initially export an empty object so external requires will get
|
||||||
|
|||||||
Reference in New Issue
Block a user