unlock: minor fixes and increase tests coverage

This commit is contained in:
Matteo Pagliazzi
2020-06-03 17:56:06 +02:00
parent 0896837528
commit 4c653aa511
3 changed files with 139 additions and 37 deletions

View File

@@ -2,13 +2,14 @@ import unlock from '../../../website/common/script/ops/unlock';
import i18n from '../../../website/common/script/i18n';
import { generateUser } from '../../helpers/common.helper';
import { NotAuthorized, BadRequest } from '../../../website/common/script/libs/errors';
import get from 'lodash/get';
describe('shared.ops.unlock', () => {
describe.only('shared.ops.unlock', () => {
let user;
const unlockPath = 'shirt.convict,shirt.cross,shirt.fire,shirt.horizon,shirt.ocean,shirt.purple,shirt.rainbow,shirt.redblue,shirt.thunder,shirt.tropical,shirt.zombie';
const unlockGearSetPath = 'items.gear.owned.headAccessory_special_bearEars,items.gear.owned.headAccessory_special_cactusEars,items.gear.owned.headAccessory_special_foxEars,items.gear.owned.headAccessory_special_lionEars,items.gear.owned.headAccessory_special_pandaEars,items.gear.owned.headAccessory_special_pigEars,items.gear.owned.headAccessory_special_tigerEars,items.gear.owned.headAccessory_special_wolfEars';
const backgroundUnlockPath = 'background.giant_florals';
const unlockCost = 1.25;
const backgroundSetUnlockPath = 'background.archery_range,background.giant_florals,background.rainbows_end';
const usersStartingGems = 50 / 4;
beforeEach(() => {
@@ -77,6 +78,56 @@ describe('shared.ops.unlock', () => {
}
});
it('returns an error if an item does not exists', done => {
try {
unlock(user, { query: { path: 'background.invalid_background' } });
} catch (err) {
expect(err).to.be.an.instanceof(BadRequest);
expect(err.message).to.equal(i18n.t('invalidUnlockSet'));
done();
}
});
it('returns an error if there are items from multiple sets', done => {
try {
unlock(user, { query: { path: 'shirt.convict,skin.0ff591' } });
} catch (err) {
expect(err).to.be.an.instanceof(BadRequest);
expect(err.message).to.equal(i18n.t('invalidUnlockSet'));
done();
}
});
it('returns an error if gear is not from the animal set', done => {
try {
unlock(user, { query: { path: 'items.gear.owned.back_mystery_202004' } });
} catch (err) {
expect(err).to.be.an.instanceof(BadRequest);
expect(err.message).to.equal(i18n.t('invalidUnlockSet'));
done();
}
});
it('returns an error if the item is free', done => {
try {
unlock(user, { query: { path: 'shirt.black' } });
} catch (err) {
expect(err).to.be.an.instanceof(BadRequest);
expect(err.message).to.equal(i18n.t('invalidUnlockSet'));
done();
}
});
it('returns an error if an item does not belong to a set (appearances)', done => {
try {
unlock(user, { query: { path: 'shirt.pink' } });
} catch (err) {
expect(err).to.be.an.instanceof(BadRequest);
expect(err.message).to.equal(i18n.t('invalidUnlockSet'));
done();
}
});
it('returns an error when user already owns items in a full set and it would be more expensive to buy the entire set', done => {
try {
// There are 11 shirts in the set, each cost 2 gems, the full set 5 gems
@@ -141,31 +192,78 @@ describe('shared.ops.unlock', () => {
expect(user.preferences.background).to.equal('');
});
it('unlocks a full set', () => {
it('unlocks a full set of appearance items', () => {
const initialShirts = Object.keys(user.purchased.shirt).length;
const [, message] = unlock(user, { query: { path: unlockPath } });
expect(message).to.equal(i18n.t('unlocked'));
expect(user.purchased.shirt.convict).to.be.true;
const individualPaths = unlockPath.split(',');
individualPaths.forEach(path => {
expect(get(user.purchased, path)).to.be.true;
});
expect(Object.keys(user.purchased.shirt).length)
.to.equal(initialShirts + individualPaths.length);
expect(user.balance).to.equal(usersStartingGems - 1.25);
});
it('unlocks a full set of gear', () => {
const initialGear = Object.keys(user.items.gear.owned).length;
const [, message] = unlock(user, { query: { path: unlockGearSetPath } });
expect(message).to.equal(i18n.t('unlocked'));
expect(user.items.gear.owned.headAccessory_special_wolfEars).to.be.true;
const individualPaths = unlockGearSetPath.split(',');
individualPaths.forEach(path => {
expect(get(user, path)).to.be.true;
});
expect(Object.keys(user.items.gear.owned).length)
.to.equal(initialGear + individualPaths.length);
expect(user.balance).to.equal(usersStartingGems - 1.25);
});
it('unlocks an item', () => {
it('unlocks a full set of backgrounds', () => {
const initialBackgrounds = Object.keys(user.purchased.background).length;
const [, message] = unlock(user, { query: { path: backgroundSetUnlockPath } });
expect(message).to.equal(i18n.t('unlocked'));
const individualPaths = backgroundSetUnlockPath.split(',');
individualPaths.forEach(path => {
expect(get(user.purchased, path)).to.be.true;
});
expect(Object.keys(user.purchased.background).length)
.to.equal(initialBackgrounds + individualPaths.length);
expect(user.balance).to.equal(usersStartingGems - 3.75);
});
it('unlocks an item (appearance)', () => {
const path = unlockPath.split(',')[0];
const initialShirts = Object.keys(user.purchased.shirt).length;
const [, message] = unlock(user, { query: { path } });
expect(message).to.equal(i18n.t('unlocked'));
expect(Object.keys(user.purchased.shirt).length).to.equal(initialShirts + 1);
expect(get(user.purchased, path)).to.be.true;
expect(user.balance).to.equal(usersStartingGems - 0.5);
});
it('unlocks an item (gear)', () => {
const path = unlockGearSetPath.split(',')[0];
const initialGear = Object.keys(user.items.gear.owned).length;
const [, message] = unlock(user, { query: { path } });
expect(message).to.equal(i18n.t('unlocked'));
expect(Object.keys(user.items.gear.owned).length).to.equal(initialGear + 1);
expect(get(user, path)).to.be.true;
expect(user.balance).to.equal(usersStartingGems - 0.5);
});
it('unlocks an item (background)', () => {
const initialBackgrounds = Object.keys(user.purchased.background).length;
const [, message] = unlock(user, { query: { path: backgroundUnlockPath } });
expect(message).to.equal(i18n.t('unlocked'));
expect(user.purchased.background.giant_florals).to.be.true;
});
it('reduces a user\'s balance', () => {
const [, message] = unlock(user, { query: { path: unlockPath } });
expect(message).to.equal(i18n.t('unlocked'));
expect(user.balance).to.equal(usersStartingGems - unlockCost);
expect(Object.keys(user.purchased.background).length).to.equal(initialBackgrounds + 1);
expect(get(user.purchased, backgroundUnlockPath)).to.be.true;
expect(user.balance).to.equal(usersStartingGems - 1.75);
});
});

View File

@@ -99,6 +99,7 @@
"unlocked": "Items have been unlocked",
"alreadyUnlocked": "Full set already unlocked.",
"alreadyUnlockedPart": "Full set already partially unlocked.",
"invalidUnlockSet": "This set of items is invalid and cannot be unlocked.",
"invalidQuantity": "Quantity to purchase must be a positive whole number.",
"USD": "(USD)",

View File

@@ -20,15 +20,14 @@ function splitPathItem (path) {
/**
* Throw an error when the provided set isn't valid.
*/
function invalidSet () {
throw new BadRequest("invalid set string");
function invalidSet (req) {
throw new BadRequest(i18n.t('invalidUnlockSet', req.language));
}
/**
* Return an item given its path and the type of set
*/
function getItemByPath (path, setType) {
console.log('getting item by path', path, setType);
const itemKey = splitPathItem(path)[1];
const item = setType === 'gear'
? content.gear.flat[itemKey]
@@ -40,27 +39,25 @@ function getItemByPath (path, setType) {
/**
* Return the type of the set (gear or one of the appareance sets - see content.appearances).
*/
function getSetType (firstPath) {
function getSetType (firstPath, req) {
if (firstPath.includes('gear.')) return 'gear';
const type = firstPath.split('.')[0];
if (content.appearances[type]) return type;
return invalidSet();
return invalidSet(req);
}
/**
* Return the set of items to unlock given the path of the first item in the set.
*/
function getSet (setType, firstPath) {
console.log('getting set from type and firstpath', setType, firstPath);
function getSet (setType, firstPath, req) {
const item = getItemByPath(firstPath, setType);
console.log('item', item);
if (!item) return invalidSet();
if (!item) return invalidSet(req);
if (setType === 'gear') {
// Only animal gear sets are unlockable
if (item.gearSet !== 'animal') return invalidSet();
if (item.gearSet !== 'animal') return invalidSet(req);
// Since each type of gear has only one purchasable set (the animal set)
// we get all items with the same type and gearSet === 'animal'
@@ -77,10 +74,9 @@ function getSet (setType, firstPath) {
return { items, paths, set: { setPrice: 5 } };
}
console.log('not a gear set');
const { set } = item;
if (!set || set.setPrice === 0) return invalidSet();
if (!set || set.setPrice === 0) return invalidSet(req);
const items = [];
const paths = [];
@@ -119,6 +115,9 @@ function markModified (user, path) {
if (user.markModified) user.markModified(path);
}
/**
* Purchase an item from a set for a given user
*/
function purchaseItem (path, setType, user) {
const isGear = setType === 'gear';
@@ -133,10 +132,13 @@ function purchaseItem (path, setType, user) {
}
}
function getIndividualItemPrice (setType, item) {
/**
* Return the price of a single item in a set
*/
function getIndividualItemPrice (setType, item, req) {
if (setType === 'gear') return 0.5;
if (!item.price || item.price === 0) return invalidSet();
if (!item.price || item.price === 0) return invalidSet(req);
return item.price / 4;
}
@@ -162,29 +164,30 @@ export default function unlock (user, req = {}, analytics) {
const isFullSet = setPaths.length > 1;
const firstPath = setPaths[0];
const setType = getSetType(firstPath);
const setType = getSetType(firstPath, req);
const isBackground = setType === 'background';
// We take the first path and use it to get the set,
// The passed paths are not used anymore after this point for full sets
const { set, items, paths } = getSet(setType, firstPath);
const { set, items, paths } = getSet(setType, firstPath, req);
let cost;
let unlockedAlready = false;
console.log('isFullSet', isFullSet, 'setType', setType);
if (isFullSet) {
console.log('fullset', {items}, {paths}, {set});
// Make sure the paths as parameters match the ones from the set
if (setPaths.length !== paths.length) return invalidSet(req);
if (!setPaths.every(setPath => paths.includes(setPath))) return invalidSet(req);
cost = set.setPrice / 4;
// all items in a set have the same price
const individualPrice = getIndividualItemPrice(setType, items[0]);
const individualPrice = getIndividualItemPrice(setType, items[0], req);
const alreadyUnlockedItems = paths
.filter(itemPath => alreadyUnlocked(user, setType, itemPath)).length;
const totalItems = items.length;
console.log('totalItems', totalItems, 'alreadyUnlockedItems', alreadyUnlockedItems)
if (alreadyUnlockedItems === totalItems) {
throw new NotAuthorized(i18n.t('alreadyUnlocked', req.language));
} else if ((totalItems - alreadyUnlockedItems) * individualPrice < cost) {
@@ -193,10 +196,10 @@ export default function unlock (user, req = {}, analytics) {
} else {
const item = getItemByPath(firstPath, setType);
if (!item || !items.includes(item) || !paths.includes(firstPath)) {
return invalidSet();
return invalidSet(req);
}
cost = getIndividualItemPrice(setType, item);
cost = getIndividualItemPrice(setType, item, req);
unlockedAlready = alreadyUnlocked(user, setType, firstPath);