mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-17 22:57:21 +01:00
Merge branch 'api-v3' into api-v3-equip-feed-hatch2
This commit is contained in:
@@ -112,6 +112,8 @@ import allocateNow from './ops/allocateNow';
|
|||||||
import hatch from './ops/hatch';
|
import hatch from './ops/hatch';
|
||||||
import feed from './ops/feed';
|
import feed from './ops/feed';
|
||||||
import equip from './ops/equip';
|
import equip from './ops/equip';
|
||||||
|
import changeClass from './ops/changeClass';
|
||||||
|
import disableClasses from './ops/disableClasses';
|
||||||
|
|
||||||
api.ops = {
|
api.ops = {
|
||||||
scoreTask,
|
scoreTask,
|
||||||
@@ -125,6 +127,8 @@ api.ops = {
|
|||||||
hatch,
|
hatch,
|
||||||
feed,
|
feed,
|
||||||
equip,
|
equip,
|
||||||
|
changeClass,
|
||||||
|
disableClasses,
|
||||||
};
|
};
|
||||||
|
|
||||||
import handleTwoHanded from './fns/handleTwoHanded';
|
import handleTwoHanded from './fns/handleTwoHanded';
|
||||||
|
|||||||
@@ -2,58 +2,70 @@ import i18n from '../i18n';
|
|||||||
import _ from 'lodash';
|
import _ from 'lodash';
|
||||||
import splitWhitespace from '../libs/splitWhitespace';
|
import splitWhitespace from '../libs/splitWhitespace';
|
||||||
import { capByLevel } from '../statHelpers';
|
import { capByLevel } from '../statHelpers';
|
||||||
|
import {
|
||||||
|
NotAuthorized,
|
||||||
|
} from '../libs/errors';
|
||||||
|
|
||||||
|
module.exports = function changeClass (user, req = {}, analytics) {
|
||||||
|
let klass = _.get(req, 'query.class');
|
||||||
|
|
||||||
module.exports = function(user, req, cb, analytics) {
|
|
||||||
var analyticsData, klass, ref;
|
|
||||||
klass = (ref = req.query) != null ? ref["class"] : void 0;
|
|
||||||
if (klass === 'warrior' || klass === 'rogue' || klass === 'wizard' || klass === 'healer') {
|
if (klass === 'warrior' || klass === 'rogue' || klass === 'wizard' || klass === 'healer') {
|
||||||
analyticsData = {
|
user.stats.class = klass;
|
||||||
|
user.flags.classSelected = true;
|
||||||
|
|
||||||
|
_.each(['weapon', 'armor', 'shield', 'head'], (type) => {
|
||||||
|
let foundKey = false;
|
||||||
|
_.findLast(user.items.gear.owned, (val, key) => {
|
||||||
|
if (key.indexOf(`${type}_${klass}`) !== -1 && val === true) {
|
||||||
|
foundKey = key;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!foundKey) {
|
||||||
|
if (type === 'weapon') {
|
||||||
|
foundKey = `weapon_${klass}_0`;
|
||||||
|
} else if (type === 'shield' && klass === 'rogue') {
|
||||||
|
foundKey = 'shield_rogue_0';
|
||||||
|
} else {
|
||||||
|
foundKey = `${type}_base_0`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
user.items.gear.equipped[type] = foundKey;
|
||||||
|
|
||||||
|
if (type === 'weapon' || (type === 'shield' && klass === 'rogue')) { // eslint-disable-line no-extra-parens
|
||||||
|
user.items.gear.owned[`${type}_${klass}_0`] = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (analytics) {
|
||||||
|
analytics.track('change class', {
|
||||||
uuid: user._id,
|
uuid: user._id,
|
||||||
"class": klass,
|
class: klass,
|
||||||
acquireMethod: 'Gems',
|
acquireMethod: 'Gems',
|
||||||
gemCost: 3,
|
gemCost: 3,
|
||||||
category: 'behavior'
|
category: 'behavior',
|
||||||
};
|
|
||||||
if (analytics != null) {
|
|
||||||
analytics.track('change class', analyticsData);
|
|
||||||
}
|
|
||||||
user.stats["class"] = klass;
|
|
||||||
user.flags.classSelected = true;
|
|
||||||
_.each(["weapon", "armor", "shield", "head"], function(type) {
|
|
||||||
var foundKey;
|
|
||||||
foundKey = false;
|
|
||||||
_.findLast(user.items.gear.owned, function(v, k) {
|
|
||||||
if (~k.indexOf(type + "_" + klass) && v === true) {
|
|
||||||
return foundKey = k;
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
user.items.gear.equipped[type] = foundKey ? foundKey : type === "weapon" ? "weapon_" + klass + "_0" : type === "shield" && klass === "rogue" ? "shield_rogue_0" : type + "_base_0";
|
|
||||||
if (type === "weapon" || (type === "shield" && klass === "rogue")) {
|
|
||||||
user.items.gear.owned[type + "_" + klass + "_0"] = true;
|
|
||||||
}
|
}
|
||||||
return true;
|
|
||||||
});
|
|
||||||
} else {
|
} else {
|
||||||
if (user.preferences.disableClasses) {
|
if (user.preferences.disableClasses) {
|
||||||
user.preferences.disableClasses = false;
|
user.preferences.disableClasses = false;
|
||||||
user.preferences.autoAllocate = false;
|
user.preferences.autoAllocate = false;
|
||||||
} else {
|
} else {
|
||||||
if (!(user.balance >= .75)) {
|
if (user.balance < 0.75) throw new NotAuthorized(i18n.t('notEnoughGems', req.language));
|
||||||
return typeof cb === "function" ? cb({
|
user.balance -= 0.75;
|
||||||
code: 401,
|
|
||||||
message: i18n.t('notEnoughGems', req.language)
|
|
||||||
}) : void 0;
|
|
||||||
}
|
}
|
||||||
user.balance -= .75;
|
|
||||||
}
|
user.stats.str = 0;
|
||||||
_.merge(user.stats, {
|
user.stats.con = 0;
|
||||||
str: 0,
|
user.stats.per = 0;
|
||||||
con: 0,
|
user.stats.int = 0;
|
||||||
per: 0,
|
user.stats.points = capByLevel(user.stats.lvl);
|
||||||
int: 0,
|
|
||||||
points: capByLevel(user.stats.lvl)
|
|
||||||
});
|
|
||||||
user.flags.classSelected = false;
|
user.flags.classSelected = false;
|
||||||
}
|
}
|
||||||
return typeof cb === "function" ? cb(null, _.pick(user, splitWhitespace('stats flags items preferences'))) : void 0;
|
|
||||||
|
return {
|
||||||
|
data: _.pick(user, splitWhitespace('stats flags items preferences')),
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -2,12 +2,15 @@ import splitWhitespace from '../libs/splitWhitespace';
|
|||||||
import { capByLevel } from '../statHelpers';
|
import { capByLevel } from '../statHelpers';
|
||||||
import _ from 'lodash';
|
import _ from 'lodash';
|
||||||
|
|
||||||
module.exports = function(user, req, cb) {
|
module.exports = function disableClasses (user) {
|
||||||
user.stats["class"] = 'warrior';
|
user.stats.class = 'warrior';
|
||||||
user.flags.classSelected = true;
|
user.flags.classSelected = true;
|
||||||
user.preferences.disableClasses = true;
|
user.preferences.disableClasses = true;
|
||||||
user.preferences.autoAllocate = true;
|
user.preferences.autoAllocate = true;
|
||||||
user.stats.str = capByLevel(user.stats.lvl);
|
user.stats.str = capByLevel(user.stats.lvl);
|
||||||
user.stats.points = 0;
|
user.stats.points = 0;
|
||||||
return typeof cb === "function" ? cb(null, _.pick(user, splitWhitespace('stats flags preferences'))) : void 0;
|
|
||||||
|
return {
|
||||||
|
data: _.pick(user, splitWhitespace('stats flags preferences')),
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -20,14 +20,12 @@ const COMMON_FILES = [
|
|||||||
'!./common/script/ops/addTask.js',
|
'!./common/script/ops/addTask.js',
|
||||||
'!./common/script/ops/addWebhook.js',
|
'!./common/script/ops/addWebhook.js',
|
||||||
'!./common/script/ops/blockUser.js',
|
'!./common/script/ops/blockUser.js',
|
||||||
'!./common/script/ops/changeClass.js',
|
|
||||||
'!./common/script/ops/clearCompleted.js',
|
'!./common/script/ops/clearCompleted.js',
|
||||||
'!./common/script/ops/clearPMs.js',
|
'!./common/script/ops/clearPMs.js',
|
||||||
'!./common/script/ops/deletePM.js',
|
'!./common/script/ops/deletePM.js',
|
||||||
'!./common/script/ops/deleteTag.js',
|
'!./common/script/ops/deleteTag.js',
|
||||||
'!./common/script/ops/deleteTask.js',
|
'!./common/script/ops/deleteTask.js',
|
||||||
'!./common/script/ops/deleteWebhook.js',
|
'!./common/script/ops/deleteWebhook.js',
|
||||||
'!./common/script/ops/disableClasses.js',
|
|
||||||
'!./common/script/ops/getTag.js',
|
'!./common/script/ops/getTag.js',
|
||||||
'!./common/script/ops/getTags.js',
|
'!./common/script/ops/getTags.js',
|
||||||
'!./common/script/ops/hourglassPurchase.js',
|
'!./common/script/ops/hourglassPurchase.js',
|
||||||
|
|||||||
27
test/api/v3/integration/user/POST-user_change-class.test.js
Normal file
27
test/api/v3/integration/user/POST-user_change-class.test.js
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
import {
|
||||||
|
generateUser,
|
||||||
|
} from '../../../../helpers/api-integration/v3';
|
||||||
|
|
||||||
|
describe('POST /user/change-class', () => {
|
||||||
|
let user;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
user = await generateUser();
|
||||||
|
});
|
||||||
|
|
||||||
|
// More tests in common code unit tests
|
||||||
|
|
||||||
|
it('changes class', async () => {
|
||||||
|
let res = await user.post(`/user/change-class?class=rogue`);
|
||||||
|
await user.sync();
|
||||||
|
|
||||||
|
expect(res).to.eql({
|
||||||
|
data: JSON.parse(JSON.stringify({
|
||||||
|
preferences: user.preferences,
|
||||||
|
stats: user.stats,
|
||||||
|
flags: user.flags,
|
||||||
|
items: user.items,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
import {
|
||||||
|
generateUser,
|
||||||
|
} from '../../../../helpers/api-integration/v3';
|
||||||
|
|
||||||
|
describe('POST /user/disable-classes', () => {
|
||||||
|
let user;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
user = await generateUser();
|
||||||
|
});
|
||||||
|
|
||||||
|
// More tests in common code unit tests
|
||||||
|
|
||||||
|
it('disable classes', async () => {
|
||||||
|
let res = await user.post(`/user/disable-classes`);
|
||||||
|
await user.sync();
|
||||||
|
|
||||||
|
expect(res).to.eql({
|
||||||
|
data: JSON.parse(JSON.stringify({
|
||||||
|
preferences: user.preferences,
|
||||||
|
stats: user.stats,
|
||||||
|
flags: user.flags,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -116,6 +116,40 @@ describe('POST /user/auth/local/register', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
context('attach to facebook user', () => {
|
||||||
|
let user;
|
||||||
|
let email = 'some@email.net';
|
||||||
|
let username = 'some-username';
|
||||||
|
let password = 'some-password';
|
||||||
|
beforeEach(async () => {
|
||||||
|
user = await generateUser();
|
||||||
|
});
|
||||||
|
it('checks onlySocialAttachLocal', async () => {
|
||||||
|
await expect(user.post('/user/auth/local/register', {
|
||||||
|
email,
|
||||||
|
username,
|
||||||
|
password,
|
||||||
|
confirmPassword: password,
|
||||||
|
})).to.eventually.be.rejected.and.eql({
|
||||||
|
code: 401,
|
||||||
|
error: 'NotAuthorized',
|
||||||
|
message: t('onlySocialAttachLocal'),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
it('succeeds', async () => {
|
||||||
|
await user.update({ 'auth.facebook.id': 'some-fb-id', 'auth.local': { ok: true } });
|
||||||
|
await user.post('/user/auth/local/register', {
|
||||||
|
username,
|
||||||
|
email,
|
||||||
|
password,
|
||||||
|
confirmPassword: password,
|
||||||
|
});
|
||||||
|
await user.sync();
|
||||||
|
expect(user.auth.local.username).to.eql(username);
|
||||||
|
expect(user.auth.local.email).to.eql(email);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
context('login is already taken', () => {
|
context('login is already taken', () => {
|
||||||
let username, email, api;
|
let username, email, api;
|
||||||
|
|
||||||
|
|||||||
119
test/common/ops/changeClass.js
Normal file
119
test/common/ops/changeClass.js
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
import changeClass from '../../../common/script/ops/changeClass';
|
||||||
|
import {
|
||||||
|
NotAuthorized,
|
||||||
|
} from '../../../common/script/libs/errors';
|
||||||
|
import i18n from '../../../common/script/i18n';
|
||||||
|
import {
|
||||||
|
generateUser,
|
||||||
|
} from '../../helpers/common.helper';
|
||||||
|
|
||||||
|
describe('shared.ops.changeClass', () => {
|
||||||
|
let user;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
user = generateUser();
|
||||||
|
});
|
||||||
|
|
||||||
|
context('req.query.class is a valid class', () => {
|
||||||
|
it('changes class', () => {
|
||||||
|
user.stats.class = 'healer';
|
||||||
|
user.items.gear.owned.armor_rogue_1 = true; // eslint-disable-line camelcase
|
||||||
|
|
||||||
|
let res = changeClass(user, {query: {class: 'rogue'}});
|
||||||
|
expect(res).to.eql({
|
||||||
|
data: {
|
||||||
|
preferences: user.preferences,
|
||||||
|
stats: user.stats,
|
||||||
|
flags: user.flags,
|
||||||
|
items: user.items,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(user.stats.class).to.equal('rogue');
|
||||||
|
expect(user.flags.classSelected).to.be.true;
|
||||||
|
expect(user.items.gear.equipped.weapon).to.equal('weapon_rogue_0');
|
||||||
|
expect(user.items.gear.owned.weapon_rogue_0).to.be.true;
|
||||||
|
expect(user.items.gear.equipped.armor).to.equal('armor_rogue_1');
|
||||||
|
expect(user.items.gear.owned.armor_rogue_1).to.be.true;
|
||||||
|
expect(user.items.gear.equipped.shield).to.equal('shield_rogue_0');
|
||||||
|
expect(user.items.gear.owned.shield_rogue_0).to.be.true;
|
||||||
|
expect(user.items.gear.equipped.head).to.equal('head_base_0');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
context('req.query.class is missing', () => {
|
||||||
|
it('has user.preferences.disableClasses === true', () => {
|
||||||
|
user.balance = 1;
|
||||||
|
user.preferences.disableClasses = true;
|
||||||
|
user.preferences.autoAllocate = true;
|
||||||
|
user.stats.points = 45;
|
||||||
|
user.stats.lvl = 3;
|
||||||
|
user.stats.str = 1;
|
||||||
|
user.stats.con = 2;
|
||||||
|
user.stats.per = 3;
|
||||||
|
user.stats.int = 4;
|
||||||
|
user.flags.classSelected = true;
|
||||||
|
|
||||||
|
let res = changeClass(user);
|
||||||
|
expect(res).to.eql({
|
||||||
|
data: {
|
||||||
|
preferences: user.preferences,
|
||||||
|
stats: user.stats,
|
||||||
|
flags: user.flags,
|
||||||
|
items: user.items,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(user.preferences.disableClasses).to.be.false;
|
||||||
|
expect(user.preferences.autoAllocate).to.be.false;
|
||||||
|
expect(user.balance).to.equal(1);
|
||||||
|
expect(user.stats.str).to.equal(0);
|
||||||
|
expect(user.stats.con).to.equal(0);
|
||||||
|
expect(user.stats.per).to.equal(0);
|
||||||
|
expect(user.stats.int).to.equal(0);
|
||||||
|
expect(user.stats.points).to.equal(3);
|
||||||
|
expect(user.flags.classSelected).to.equal(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
context('has user.preferences.disableClasses !== true', () => {
|
||||||
|
it('and less than 3 gems', () => {
|
||||||
|
user.balance = 0.5;
|
||||||
|
try {
|
||||||
|
changeClass(user);
|
||||||
|
} catch (err) {
|
||||||
|
expect(err).to.be.an.instanceof(NotAuthorized);
|
||||||
|
expect(err.message).to.equal(i18n.t('notEnoughGems'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('and at least 3 gems', () => {
|
||||||
|
user.balance = 1;
|
||||||
|
user.stats.points = 45;
|
||||||
|
user.stats.lvl = 3;
|
||||||
|
user.stats.str = 1;
|
||||||
|
user.stats.con = 2;
|
||||||
|
user.stats.per = 3;
|
||||||
|
user.stats.int = 4;
|
||||||
|
user.flags.classSelected = true;
|
||||||
|
|
||||||
|
let res = changeClass(user);
|
||||||
|
expect(res).to.eql({
|
||||||
|
data: {
|
||||||
|
preferences: user.preferences,
|
||||||
|
stats: user.stats,
|
||||||
|
flags: user.flags,
|
||||||
|
items: user.items,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(user.balance).to.equal(0.25);
|
||||||
|
expect(user.stats.str).to.equal(0);
|
||||||
|
expect(user.stats.con).to.equal(0);
|
||||||
|
expect(user.stats.per).to.equal(0);
|
||||||
|
expect(user.stats.int).to.equal(0);
|
||||||
|
expect(user.stats.points).to.equal(3);
|
||||||
|
expect(user.flags.classSelected).to.equal(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
37
test/common/ops/disableClasses.js
Normal file
37
test/common/ops/disableClasses.js
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
import disableClasses from '../../../common/script/ops/disableClasses';
|
||||||
|
import {
|
||||||
|
generateUser,
|
||||||
|
} from '../../helpers/common.helper';
|
||||||
|
|
||||||
|
describe('shared.ops.disableClasses', () => {
|
||||||
|
let user;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
user = generateUser();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('disable classes', () => {
|
||||||
|
user.stats.lvl = 34;
|
||||||
|
user.stats.str = 45;
|
||||||
|
user.stats.class = 'healer';
|
||||||
|
user.preferences.disableClasses = false;
|
||||||
|
user.preferences.autoAllocate = false;
|
||||||
|
user.stats.points = 2;
|
||||||
|
|
||||||
|
let res = disableClasses(user);
|
||||||
|
expect(res).to.eql({
|
||||||
|
data: {
|
||||||
|
preferences: user.preferences,
|
||||||
|
stats: user.stats,
|
||||||
|
flags: user.flags,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(user.stats.class).to.equal('warrior');
|
||||||
|
expect(user.flags.classSelected).to.equal(true);
|
||||||
|
expect(user.preferences.disableClasses).to.equal(true);
|
||||||
|
expect(user.preferences.autoAllocate).to.equal(true);
|
||||||
|
expect(user.stats.str).to.equal(34);
|
||||||
|
expect(user.stats.points).to.equal(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -70,7 +70,7 @@ api.registerLocal = {
|
|||||||
url: '/user/auth/local/register',
|
url: '/user/auth/local/register',
|
||||||
async handler (req, res) {
|
async handler (req, res) {
|
||||||
let fbUser = res.locals.user; // If adding local auth to social user
|
let fbUser = res.locals.user; // If adding local auth to social user
|
||||||
// TODO check user doesn't have local auth
|
|
||||||
req.checkBody({
|
req.checkBody({
|
||||||
email: {
|
email: {
|
||||||
notEmpty: {errorMessage: res.t('missingEmail')},
|
notEmpty: {errorMessage: res.t('missingEmail')},
|
||||||
@@ -82,7 +82,6 @@ api.registerLocal = {
|
|||||||
equals: {options: [req.body.confirmPassword], errorMessage: res.t('passwordConfirmationMatch')},
|
equals: {options: [req.body.confirmPassword], errorMessage: res.t('passwordConfirmationMatch')},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
let validationErrors = req.validationErrors();
|
let validationErrors = req.validationErrors();
|
||||||
if (validationErrors) throw validationErrors;
|
if (validationErrors) throw validationErrors;
|
||||||
|
|
||||||
@@ -124,7 +123,7 @@ api.registerLocal = {
|
|||||||
|
|
||||||
if (fbUser) {
|
if (fbUser) {
|
||||||
if (!fbUser.auth.facebook.id) throw new NotAuthorized(res.t('onlySocialAttachLocal'));
|
if (!fbUser.auth.facebook.id) throw new NotAuthorized(res.t('onlySocialAttachLocal'));
|
||||||
fbUser.auth.local = newUser;
|
fbUser.auth.local = newUser.auth.local;
|
||||||
newUser = fbUser;
|
newUser = fbUser;
|
||||||
} else {
|
} else {
|
||||||
newUser = new User(newUser);
|
newUser = new User(newUser);
|
||||||
|
|||||||
@@ -381,4 +381,46 @@ api.feed = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @api {post} /user/change-class Change class.
|
||||||
|
* @apiVersion 3.0.0
|
||||||
|
* @apiName UserChangeClass
|
||||||
|
* @apiGroup User
|
||||||
|
*
|
||||||
|
* @apiParam {string} class ?class={warrior|rogue|wizard|healer}. If missing will
|
||||||
|
*
|
||||||
|
* @apiSuccess {Object} data `stats flags items preferences`
|
||||||
|
*/
|
||||||
|
api.changeClass = {
|
||||||
|
method: 'POST',
|
||||||
|
middlewares: [authWithHeaders(), cron],
|
||||||
|
url: '/user/change-class',
|
||||||
|
async handler (req, res) {
|
||||||
|
let user = res.locals.user;
|
||||||
|
let changeClassRes = common.ops.changeClass(user, req, res.analytics);
|
||||||
|
await user.save();
|
||||||
|
res.respond(200, changeClassRes);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @api {post} /user/disable-classes Disable classes.
|
||||||
|
* @apiVersion 3.0.0
|
||||||
|
* @apiName UserDisableClasses
|
||||||
|
* @apiGroup User
|
||||||
|
*
|
||||||
|
* @apiSuccess {Object} data `stats flags preferences`
|
||||||
|
*/
|
||||||
|
api.disableClasses = {
|
||||||
|
method: 'POST',
|
||||||
|
middlewares: [authWithHeaders(), cron],
|
||||||
|
url: '/user/disable-classes',
|
||||||
|
async handler (req, res) {
|
||||||
|
let user = res.locals.user;
|
||||||
|
let disableClassesRes = common.ops.disableClasses(user, req);
|
||||||
|
await user.save();
|
||||||
|
res.respond(200, disableClassesRes);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = api;
|
module.exports = api;
|
||||||
|
|||||||
Reference in New Issue
Block a user