mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-16 06:07:21 +01:00
fix buying transformation items
This commit is contained in:
@@ -1,89 +0,0 @@
|
|||||||
import { BuySpellOperation } from '../../../../website/common/script/ops/buy/buySpell';
|
|
||||||
import {
|
|
||||||
BadRequest,
|
|
||||||
NotFound,
|
|
||||||
NotAuthorized,
|
|
||||||
} from '../../../../website/common/script/libs/errors';
|
|
||||||
import i18n from '../../../../website/common/script/i18n';
|
|
||||||
import {
|
|
||||||
generateUser,
|
|
||||||
} from '../../../helpers/common.helper';
|
|
||||||
import content from '../../../../website/common/script/content/index';
|
|
||||||
import { errorMessage } from '../../../../website/common/script/libs/errorMessage';
|
|
||||||
|
|
||||||
describe('shared.ops.buySpecialSpell', () => {
|
|
||||||
let user;
|
|
||||||
const analytics = { track () {} };
|
|
||||||
|
|
||||||
async function buySpecialSpell (_user, _req, _analytics) {
|
|
||||||
const buyOp = new BuySpellOperation(_user, _req, _analytics);
|
|
||||||
|
|
||||||
return buyOp.purchase();
|
|
||||||
}
|
|
||||||
beforeEach(() => {
|
|
||||||
user = generateUser();
|
|
||||||
sinon.stub(analytics, 'track');
|
|
||||||
});
|
|
||||||
|
|
||||||
afterEach(() => {
|
|
||||||
analytics.track.restore();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('throws an error if params.key is missing', async () => {
|
|
||||||
try {
|
|
||||||
await buySpecialSpell(user);
|
|
||||||
} catch (err) {
|
|
||||||
expect(err).to.be.an.instanceof(BadRequest);
|
|
||||||
expect(err.message).to.equal(errorMessage('missingKeyParam'));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
it('throws an error if the spell doesn\'t exists', async () => {
|
|
||||||
try {
|
|
||||||
await buySpecialSpell(user, {
|
|
||||||
params: {
|
|
||||||
key: 'notExisting',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} catch (err) {
|
|
||||||
expect(err).to.be.an.instanceof(NotFound);
|
|
||||||
expect(err.message).to.equal(errorMessage('spellNotFound', { spellId: 'notExisting' }));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
it('throws an error if the user doesn\'t have enough gold', async () => {
|
|
||||||
user.stats.gp = 1;
|
|
||||||
try {
|
|
||||||
await buySpecialSpell(user, {
|
|
||||||
params: {
|
|
||||||
key: 'thankyou',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} catch (err) {
|
|
||||||
expect(err).to.be.an.instanceof(NotAuthorized);
|
|
||||||
expect(err.message).to.equal(i18n.t('messageNotEnoughGold'));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
it('buys an item', async () => {
|
|
||||||
user.stats.gp = 11;
|
|
||||||
const item = content.special.thankyou;
|
|
||||||
|
|
||||||
const [data, message] = await buySpecialSpell(user, {
|
|
||||||
params: {
|
|
||||||
key: 'thankyou',
|
|
||||||
},
|
|
||||||
}, analytics);
|
|
||||||
|
|
||||||
expect(user.stats.gp).to.equal(1);
|
|
||||||
expect(user.items.special.thankyou).to.equal(1);
|
|
||||||
expect(data).to.eql({
|
|
||||||
items: user.items,
|
|
||||||
stats: user.stats,
|
|
||||||
});
|
|
||||||
expect(message).to.equal(i18n.t('messageBought', {
|
|
||||||
itemText: item.text(),
|
|
||||||
}));
|
|
||||||
expect(analytics.track).to.be.calledOnce;
|
|
||||||
});
|
|
||||||
});
|
|
||||||
172
test/common/ops/buy/buySpell.test.js
Normal file
172
test/common/ops/buy/buySpell.test.js
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
import { BuySpellOperation } from '../../../../website/common/script/ops/buy/buySpell';
|
||||||
|
import {
|
||||||
|
BadRequest,
|
||||||
|
NotFound,
|
||||||
|
NotAuthorized,
|
||||||
|
} from '../../../../website/common/script/libs/errors';
|
||||||
|
import i18n from '../../../../website/common/script/i18n';
|
||||||
|
import {
|
||||||
|
generateUser,
|
||||||
|
} from '../../../helpers/common.helper';
|
||||||
|
import content from '../../../../website/common/script/content/index';
|
||||||
|
import { errorMessage } from '../../../../website/common/script/libs/errorMessage';
|
||||||
|
|
||||||
|
describe('shared.ops.buySpecialSpell', () => {
|
||||||
|
let user;
|
||||||
|
let clock;
|
||||||
|
const analytics = { track () {} };
|
||||||
|
|
||||||
|
async function buySpecialSpell (_user, _req, _analytics) {
|
||||||
|
const buyOp = new BuySpellOperation(_user, _req, _analytics);
|
||||||
|
|
||||||
|
return buyOp.purchase();
|
||||||
|
}
|
||||||
|
beforeEach(() => {
|
||||||
|
user = generateUser();
|
||||||
|
sinon.stub(analytics, 'track');
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
analytics.track.restore();
|
||||||
|
if (clock) {
|
||||||
|
clock.restore();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('throws an error if params.key is missing', async () => {
|
||||||
|
try {
|
||||||
|
await buySpecialSpell(user);
|
||||||
|
} catch (err) {
|
||||||
|
expect(err).to.be.an.instanceof(BadRequest);
|
||||||
|
expect(err.message).to.equal(errorMessage('missingKeyParam'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('throws an error if the item doesn\'t exists', async () => {
|
||||||
|
try {
|
||||||
|
await buySpecialSpell(user, {
|
||||||
|
params: {
|
||||||
|
key: 'notExisting',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
expect(err).to.be.an.instanceof(NotFound);
|
||||||
|
expect(err.message).to.equal(errorMessage('spellNotFound', { spellId: 'notExisting' }));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('throws an error if the user doesn\'t have enough gold', async () => {
|
||||||
|
user.stats.gp = 1;
|
||||||
|
try {
|
||||||
|
await buySpecialSpell(user, {
|
||||||
|
params: {
|
||||||
|
key: 'thankyou',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
expect(err).to.be.an.instanceof(NotAuthorized);
|
||||||
|
expect(err.message).to.equal(i18n.t('messageNotEnoughGold'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('buying cards', () => {
|
||||||
|
it('buys a card that is always available', async () => {
|
||||||
|
user.stats.gp = 11;
|
||||||
|
const item = content.special.thankyou;
|
||||||
|
|
||||||
|
const [data, message] = await buySpecialSpell(user, {
|
||||||
|
params: {
|
||||||
|
key: 'thankyou',
|
||||||
|
},
|
||||||
|
}, analytics);
|
||||||
|
|
||||||
|
expect(user.stats.gp).to.equal(1);
|
||||||
|
expect(user.items.special.thankyou).to.equal(1);
|
||||||
|
expect(data).to.eql({
|
||||||
|
items: user.items,
|
||||||
|
stats: user.stats,
|
||||||
|
});
|
||||||
|
expect(message).to.equal(i18n.t('messageBought', {
|
||||||
|
itemText: item.text(),
|
||||||
|
}));
|
||||||
|
expect(analytics.track).to.be.calledOnce;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('buys a limited card when it is available', async () => {
|
||||||
|
user.stats.gp = 11;
|
||||||
|
const item = content.special.nye;
|
||||||
|
clock = sinon.useFakeTimers(new Date('2024-01-01'));
|
||||||
|
|
||||||
|
const [data, message] = await buySpecialSpell(user, {
|
||||||
|
params: {
|
||||||
|
key: 'nye',
|
||||||
|
},
|
||||||
|
}, analytics);
|
||||||
|
|
||||||
|
expect(user.stats.gp).to.equal(1);
|
||||||
|
expect(user.items.special.nye).to.equal(1);
|
||||||
|
expect(data).to.eql({
|
||||||
|
items: user.items,
|
||||||
|
stats: user.stats,
|
||||||
|
});
|
||||||
|
expect(message).to.equal(i18n.t('messageBought', {
|
||||||
|
itemText: item.text(),
|
||||||
|
}));
|
||||||
|
expect(analytics.track).to.be.calledOnce;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('throws an error if the card is not currently available', async () => {
|
||||||
|
user.stats.gp = 11;
|
||||||
|
clock = sinon.useFakeTimers(new Date('2024-06-01'));
|
||||||
|
try {
|
||||||
|
await buySpecialSpell(user, {
|
||||||
|
params: {
|
||||||
|
key: 'nye',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
expect(err).to.be.an.instanceof(NotAuthorized);
|
||||||
|
expect(err.message).to.equal(i18n.t('cannotBuyItem'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('buying spells', () => {
|
||||||
|
it('buys a spell if it is currently available', async () => {
|
||||||
|
user.stats.gp = 16;
|
||||||
|
clock = sinon.useFakeTimers(new Date('2024-06-22'));
|
||||||
|
const item = content.special.seafoam;
|
||||||
|
const [data, message] = await buySpecialSpell(user, {
|
||||||
|
params: {
|
||||||
|
key: 'seafoam',
|
||||||
|
},
|
||||||
|
}, analytics);
|
||||||
|
|
||||||
|
expect(user.stats.gp).to.equal(1);
|
||||||
|
expect(user.items.special.seafoam).to.equal(1);
|
||||||
|
expect(data).to.eql({
|
||||||
|
items: user.items,
|
||||||
|
stats: user.stats,
|
||||||
|
});
|
||||||
|
expect(message).to.equal(i18n.t('messageBought', {
|
||||||
|
itemText: item.text(),
|
||||||
|
}));
|
||||||
|
expect(analytics.track).to.be.calledOnce;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('throws an error if the spell is not currently available', async () => {
|
||||||
|
user.stats.gp = 50;
|
||||||
|
clock = sinon.useFakeTimers(new Date('2024-01-22'));
|
||||||
|
try {
|
||||||
|
await buySpecialSpell(user, {
|
||||||
|
params: {
|
||||||
|
key: 'seafoam',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
expect(err).to.be.an.instanceof(NotAuthorized);
|
||||||
|
expect(err.message).to.equal(i18n.t('cannotBuyItem'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -280,9 +280,7 @@ spells.special = {
|
|||||||
previousPurchase: true,
|
previousPurchase: true,
|
||||||
target: 'user',
|
target: 'user',
|
||||||
notes: t('spellSpecialSnowballAuraNotes'),
|
notes: t('spellSpecialSnowballAuraNotes'),
|
||||||
canOwn () {
|
limited: true,
|
||||||
return moment().isBetween('2024-01-09T08:00-05:00', EVENTS.winter2024.end);
|
|
||||||
},
|
|
||||||
cast (user, target, req) {
|
cast (user, target, req) {
|
||||||
if (!user.items.special.snowball) throw new NotAuthorized(t('spellNotOwned')(req.language));
|
if (!user.items.special.snowball) throw new NotAuthorized(t('spellNotOwned')(req.language));
|
||||||
target.stats.buffs.snowball = true;
|
target.stats.buffs.snowball = true;
|
||||||
@@ -316,9 +314,7 @@ spells.special = {
|
|||||||
previousPurchase: true,
|
previousPurchase: true,
|
||||||
target: 'user',
|
target: 'user',
|
||||||
notes: t('spellSpecialSpookySparklesNotes'),
|
notes: t('spellSpecialSpookySparklesNotes'),
|
||||||
canOwn () {
|
limited: true,
|
||||||
return moment().isBetween('2023-10-03T08:00-04:00', EVENTS.fall2023.end);
|
|
||||||
},
|
|
||||||
cast (user, target, req) {
|
cast (user, target, req) {
|
||||||
if (!user.items.special.spookySparkles) throw new NotAuthorized(t('spellNotOwned')(req.language));
|
if (!user.items.special.spookySparkles) throw new NotAuthorized(t('spellNotOwned')(req.language));
|
||||||
target.stats.buffs.snowball = false;
|
target.stats.buffs.snowball = false;
|
||||||
@@ -352,10 +348,7 @@ spells.special = {
|
|||||||
previousPurchase: true,
|
previousPurchase: true,
|
||||||
target: 'user',
|
target: 'user',
|
||||||
notes: t('spellSpecialShinySeedNotes'),
|
notes: t('spellSpecialShinySeedNotes'),
|
||||||
event: EVENTS.spring2024,
|
limited: true,
|
||||||
canOwn () {
|
|
||||||
return moment().isBetween('2024-04-18T08:00-05:00', EVENTS.spring2024.end);
|
|
||||||
},
|
|
||||||
cast (user, target, req) {
|
cast (user, target, req) {
|
||||||
if (!user.items.special.shinySeed) throw new NotAuthorized(t('spellNotOwned')(req.language));
|
if (!user.items.special.shinySeed) throw new NotAuthorized(t('spellNotOwned')(req.language));
|
||||||
target.stats.buffs.snowball = false;
|
target.stats.buffs.snowball = false;
|
||||||
@@ -389,9 +382,7 @@ spells.special = {
|
|||||||
previousPurchase: true,
|
previousPurchase: true,
|
||||||
target: 'user',
|
target: 'user',
|
||||||
notes: t('spellSpecialSeafoamNotes'),
|
notes: t('spellSpecialSeafoamNotes'),
|
||||||
canOwn () {
|
limited: true,
|
||||||
return moment().isBetween('2023-07-11T08:00-04:00', EVENTS.summer2023.end);
|
|
||||||
},
|
|
||||||
cast (user, target, req) {
|
cast (user, target, req) {
|
||||||
if (!user.items.special.seafoam) throw new NotAuthorized(t('spellNotOwned')(req.language));
|
if (!user.items.special.seafoam) throw new NotAuthorized(t('spellNotOwned')(req.language));
|
||||||
target.stats.buffs.snowball = false;
|
target.stats.buffs.snowball = false;
|
||||||
@@ -426,9 +417,7 @@ spells.special = {
|
|||||||
silent: true,
|
silent: true,
|
||||||
target: 'user',
|
target: 'user',
|
||||||
notes: t('nyeCardNotes'),
|
notes: t('nyeCardNotes'),
|
||||||
canOwn () {
|
limited: true,
|
||||||
return moment().isBetween(EVENTS.nye2023.start, EVENTS.nye2023.end);
|
|
||||||
},
|
|
||||||
cast (user, target) {
|
cast (user, target) {
|
||||||
if (user === target) {
|
if (user === target) {
|
||||||
if (!user.achievements.nye) user.achievements.nye = 0;
|
if (!user.achievements.nye) user.achievements.nye = 0;
|
||||||
@@ -466,9 +455,7 @@ spells.special = {
|
|||||||
silent: true,
|
silent: true,
|
||||||
target: 'user',
|
target: 'user',
|
||||||
notes: t('valentineCardNotes'),
|
notes: t('valentineCardNotes'),
|
||||||
canOwn () {
|
limited: true,
|
||||||
return moment().isBetween(EVENTS.valentine2024.start, EVENTS.valentine2024.end);
|
|
||||||
},
|
|
||||||
cast (user, target) {
|
cast (user, target) {
|
||||||
if (user === target) {
|
if (user === target) {
|
||||||
if (!user.achievements.valentine) user.achievements.valentine = 0;
|
if (!user.achievements.valentine) user.achievements.valentine = 0;
|
||||||
|
|||||||
@@ -8,6 +8,8 @@ import {
|
|||||||
} from '../../libs/errors';
|
} from '../../libs/errors';
|
||||||
import { AbstractGoldItemOperation } from './abstractBuyOperation';
|
import { AbstractGoldItemOperation } from './abstractBuyOperation';
|
||||||
import { errorMessage } from '../../libs/errorMessage';
|
import { errorMessage } from '../../libs/errorMessage';
|
||||||
|
import { NotAuthorized } from '../../../../server/libs/errors';
|
||||||
|
import { getScheduleMatchingGroup } from '../../content/constants/schedule';
|
||||||
|
|
||||||
export class BuySpellOperation extends AbstractGoldItemOperation { // eslint-disable-line import/prefer-default-export, max-len
|
export class BuySpellOperation extends AbstractGoldItemOperation { // eslint-disable-line import/prefer-default-export, max-len
|
||||||
getItemKey () {
|
getItemKey () {
|
||||||
@@ -22,6 +24,23 @@ export class BuySpellOperation extends AbstractGoldItemOperation { // eslint-dis
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
canUserPurchase (user, item) {
|
||||||
|
super.canUserPurchase(user, item);
|
||||||
|
|
||||||
|
if (item.limited) {
|
||||||
|
let matcherGroup;
|
||||||
|
if (content.cardTypes[item.key]) {
|
||||||
|
matcherGroup = 'card';
|
||||||
|
} else {
|
||||||
|
matcherGroup = 'seasonalSpells';
|
||||||
|
}
|
||||||
|
const matcher = getScheduleMatchingGroup(matcherGroup);
|
||||||
|
if (!matcher.match(item.key)) {
|
||||||
|
throw new NotAuthorized(this.i18n('cannotBuyItem'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
extractAndValidateParams (user, req) {
|
extractAndValidateParams (user, req) {
|
||||||
this.key = get(req, 'params.key');
|
this.key = get(req, 'params.key');
|
||||||
const { key } = this;
|
const { key } = this;
|
||||||
|
|||||||
Reference in New Issue
Block a user