initial commit

This commit is contained in:
2021-07-18 13:57:08 -06:00
commit f258a8e619
9 changed files with 1290 additions and 0 deletions

1
.gitignore vendored Normal file
View File

@ -0,0 +1 @@
node_modules

237
build/mockProcess.js Normal file
View File

@ -0,0 +1,237 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const mockdata_1 = require("./mockdata");
const cryptoRandomString = require("crypto-random-string");
const unique_names_generator_1 = require("unique-names-generator");
function handleEvent(eventLog) {
if (eventLog.eventName === 'bought') {
let vendingMachine = findVendingMachine(eventLog.data.offer);
let selection = vendingMachine.selections.find((sel) => {
return sel.id === eventLog.data.selection;
});
selection.current = selection.current - 1;
}
}
function findVendingMachine(id) {
for (let i in mockdata_1.default.players) {
let player = mockdata_1.default.players[i];
for (let j in player.vendingMachines) {
if (player.vendingMachines[i].id === id) {
return player.vendingMachines[i];
}
}
}
}
function personPassesByVendingMachine(person, machine) {
let isThursty = Math.random() < person.thirst;
let availableSelections = machine.selections.filter((selection) => {
return !!selection.current;
});
if (isThursty) {
if (!availableSelections.length) {
return wontBuy('NoneAvailable');
}
let acceptableOptions = availableSelections.filter((option) => {
return person.preferences.indexOf(option.drink) !== -1;
});
if (!acceptableOptions.length) {
return wontBuy('NoPreferredDrinks');
}
let options = acceptableOptions.filter((selection) => {
return selection.price <= person.highestPrice;
});
if (!options.length) {
return wontBuy('HighPrice');
}
let picked = pick(acceptableOptions);
let selectionTypeData = getSelectionType(picked.type);
return {
eventName: 'bought',
data: {
reason: 'Thirsty',
amount: picked.price,
drink: picked.drink,
selection: picked.id,
selectionType: picked.type,
oz: selectionTypeData ? selectionTypeData.oz : 12,
person: person.id,
offer: machine.id,
}
};
}
else {
return wontBuy('NotThirsty');
}
function wontBuy(reason) {
return {
eventName: 'wontBuy',
data: {
reason: reason,
amount: 0,
drink: 'none',
selection: 'none',
selectionType: 'none',
oz: 0,
person: person.id,
offer: machine.id,
}
};
}
}
function generatePerson(location) {
let money = Math.random() * 100;
return {
id: genId('person'),
name: genPersonName(),
preferences: genPreferences(),
highestPrice: (money / 20),
money: money,
thirst: clamp(location.dailyTraffic.thirstBonus + (Math.random() / 10))
};
}
function genId(prefix) {
return `${prefix}-${cryptoRandomString({ length: 6, type: 'distinguishable' })}`;
}
function genPersonName() {
const config = {
dictionaries: [unique_names_generator_1.names, unique_names_generator_1.names],
length: 2,
separator: ' '
};
return unique_names_generator_1.uniqueNamesGenerator(config);
}
function genDrinkName() {
const config = {
dictionaries: [unique_names_generator_1.adjectives, unique_names_generator_1.colors],
length: 2,
separator: ' ',
style: 'capital'
};
return unique_names_generator_1.uniqueNamesGenerator(config);
}
function genStoreName() {
const stores = [
'shop',
'reseller',
'department Store',
'chain Store',
'emporium',
'supermarket',
'hypermarket',
'superstore',
'mart',
'shed',
'big Box',
'convenience Store',
'corner Store',
'food Store',
'market',
'food Mart',
'bodega',
];
const config = {
dictionaries: [unique_names_generator_1.names, stores],
length: 2,
separator: '\'s ',
style: 'capital'
};
return unique_names_generator_1.uniqueNamesGenerator(config);
}
function genCityName() {
const places = [
'town',
'city',
'township',
'borough',
'seat',
'metropolis',
'burg',
'municipality',
'hamlet',
];
const config = {
dictionaries: [Math.random() > 0.5 ? unique_names_generator_1.colors : unique_names_generator_1.names, places],
length: 2,
separator: ' ',
style: 'capital'
};
return unique_names_generator_1.uniqueNamesGenerator(config);
}
function clamp(value, max = 1, min = 0) {
if (value > max) {
return max;
}
if (value < min) {
return min;
}
return value;
}
function genPreferences() {
const availableDrinks = mockdata_1.default.drinks;
let prefCount = Math.floor(Math.random() * availableDrinks.length);
prefCount = prefCount === 0 ? prefCount + 1 : prefCount;
let preferences = [];
while (preferences.length < prefCount) {
let newPref = pick(availableDrinks);
if (preferences.indexOf(newPref.id) === -1) {
preferences.push(newPref.id);
}
}
return preferences;
}
function getSelectionType(typeName) {
return mockdata_1.default.selectionTypes.find((type) => {
return type.type === typeName;
});
}
function randomNumber(min, max) {
return Math.random() * (max - min) + min;
}
function pick(arr) {
return arr[Math.floor(Math.random() * arr.length)];
}
function getStatsFromEvents(events) {
let purchaseCount = 0;
let offerEventCount = 0;
let amountEarned = 0;
let ozSold = 0;
let popularity = {};
let eventReasons = {};
for (let i in events) {
let eventLog = events[i];
if (!eventReasons[eventLog.data.reason]) {
eventReasons[eventLog.data.reason] = 0;
}
eventReasons[eventLog.data.reason] = eventReasons[eventLog.data.reason] + 1;
if (eventLog.eventName === 'bought' || eventLog.eventName === 'wontBuy') {
offerEventCount = offerEventCount + 1;
if (eventLog.eventName === 'bought') {
purchaseCount = purchaseCount + 1;
if (!popularity[eventLog.data.drink]) {
popularity[eventLog.data.drink] = 0;
}
popularity[eventLog.data.drink] = popularity[eventLog.data.drink] + 1;
amountEarned = amountEarned + eventLog.data.amount;
ozSold = ozSold + eventLog.data.oz;
}
}
}
console.log({
purchaseCount,
purchaseRate: purchaseCount / offerEventCount,
offerEventCount,
amountEarned,
ozSold,
popularity,
eventReasons
});
}
let events = [];
let traffic = randomNumber(mockdata_1.default.locations[0].dailyTraffic.min, mockdata_1.default.locations[0].dailyTraffic.max);
for (let i = 0; i < traffic; i++) {
let curEvent = personPassesByVendingMachine(generatePerson(mockdata_1.default.locations[0]), mockdata_1.default.players[0].vendingMachines[0]);
console.log(genCityName());
events.push(curEvent);
handleEvent(curEvent);
}
getStatsFromEvents(events);

220
build/mockdata.js Normal file
View File

@ -0,0 +1,220 @@
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
let dataDefinition = {
players: [
{
id: 'player-01',
name: 'payne8',
balance: 4000.00,
vendingMachines: [
{
id: 'vend-01',
name: "First One!",
location: 'loc-01',
placement: 'plac-01',
compatibleSelectionTypes: ['can', 'bottle'],
selections: [
{
id: 'sel-01',
name: 'Diet Mtn. Dew',
drink: 'drink-01',
max: 56,
current: 24,
price: 3.00,
type: 'can'
},
{
id: 'sel-02',
name: 'Mtn. Dew',
drink: 'drink-02',
max: 56,
current: 24,
price: 2.00,
type: 'can'
},
{
id: 'sel-03',
name: 'Dr. Pepper',
drink: 'drink-03',
max: 56,
current: 24,
price: 1.50,
type: 'can'
},
{
id: 'sel-04',
name: 'Sprite',
drink: 'drink-04',
max: 56,
current: 24,
price: 0.75,
type: 'can'
},
{
id: 'sel-05',
name: 'Coke',
drink: 'drink-05',
max: 56,
current: 24,
price: 1.00,
type: 'can'
},
{
id: 'sel-06',
name: 'Diet Coke',
drink: 'drink-06',
max: 56,
current: 24,
price: 1.50,
type: 'can'
},
{
id: 'sel-07',
name: 'Orange Soda',
drink: 'drink-07',
max: 56,
current: 24,
price: 2.00,
type: 'can'
},
{
id: 'sel-08',
name: 'Grape Soda',
drink: 'drink-08',
max: 56,
current: 24,
price: 0.50,
type: 'can'
},
],
agreement: 0.0,
}
],
deals: [
{
id: 'deal-01',
location: 'loc-01',
selections: [
{
id: 'sel-09',
name: 'Diet Mtn. Dew - Dozen Cans',
drink: 'drink-01',
max: 56,
current: 24,
price: 7.00,
type: 'dozen-box'
},
],
agreement: 0.4,
},
]
}
],
selectionTypes: [
{
type: 'can',
oz: 12,
},
{
type: 'dozen-box',
os: 144
}
],
drinks: [
{
id: 'drink-01',
name: 'Diet Mtn. Dew',
costPerOz: 0.005,
},
{
id: 'drink-02',
name: 'Mtn. Dew',
costPerOz: 0.005,
},
{
id: 'drink-03',
name: 'Dr. Pepper',
costPerOz: 0.005,
},
{
id: 'drink-04',
name: 'Sprite',
costPerOz: 0.005,
},
{
id: 'drink-05',
name: 'Coke',
costPerOz: 0.005,
},
{
id: 'drink-06',
name: 'Diet Coke',
costPerOz: 0.005,
},
{
id: 'drink-07',
name: 'Orange Soda',
costPerOz: 0.005,
},
{
id: 'drink-08',
name: 'Grape Soda',
costPerOz: 0.005,
},
],
locations: [
{
id: 'loc-01',
name: 'General Store',
city: 'city-01',
dailyTraffic: {
max: 2000,
min: 400,
thirstBonus: 0.05,
},
deals: {
count: 2,
traffic: 0.4,
list: [
'deal-01'
]
},
placements: [
{
id: 'plac-01',
traffic: 0.5,
},
{
id: 'plac-02',
traffic: 0.6,
},
]
}
],
cities: [
{
id: 'city-01',
name: 'Hometown',
population: 15000
}
],
people: [
{
id: 'person-01',
name: 'Steve Klemm',
preferences: [
'drink-01',
'drink-04',
'drink-08',
],
highestPrice: 2.00,
wallet: {
cc: 500.00,
cash: 10.00,
coins: 1.25,
},
thirst: 0.8,
}
]
};
exports.default = dataDefinition;

236
data-relationships.svg Normal file
View File

@ -0,0 +1,236 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<!-- Created with Inkscape (http://www.inkscape.org/) -->
<svg
xmlns:dc="http://purl.org/dc/elements/1.1/"
xmlns:cc="http://creativecommons.org/ns#"
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
xmlns:svg="http://www.w3.org/2000/svg"
xmlns="http://www.w3.org/2000/svg"
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
width="210mm"
height="297mm"
viewBox="0 0 210 297"
version="1.1"
id="svg8"
inkscape:version="0.92.4 (5da689c313, 2019-01-14)"
sodipodi:docname="data-relationships.svg">
<defs
id="defs2" />
<sodipodi:namedview
id="base"
pagecolor="#ffffff"
bordercolor="#666666"
borderopacity="1.0"
inkscape:pageopacity="0.0"
inkscape:pageshadow="2"
inkscape:zoom="1.979899"
inkscape:cx="451.62995"
inkscape:cy="862.03561"
inkscape:document-units="mm"
inkscape:current-layer="layer1"
showgrid="false"
inkscape:window-width="3440"
inkscape:window-height="1377"
inkscape:window-x="3832"
inkscape:window-y="-8"
inkscape:window-maximized="1" />
<metadata
id="metadata5">
<rdf:RDF>
<cc:Work
rdf:about="">
<dc:format>image/svg+xml</dc:format>
<dc:type
rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
<dc:title></dc:title>
</cc:Work>
</rdf:RDF>
</metadata>
<g
inkscape:label="Layer 1"
inkscape:groupmode="layer"
id="layer1">
<path
style="fill:none;stroke:#000000;stroke-width:0.26458332px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1"
d="m 35.090844,51.835306 45.758187,-4.06414"
id="path956"
inkscape:connector-curvature="0"
sodipodi:nodetypes="cc" />
<path
style="fill:none;stroke:#000000;stroke-width:0.26458332px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1"
d="m 129.22481,48.038434 49.04396,18.842501"
id="path958"
inkscape:connector-curvature="0"
sodipodi:nodetypes="cc" />
<path
style="fill:none;stroke:#000000;stroke-width:0.26458332px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1"
d="M 35.090844,51.835306 67.446288,76.541908"
id="path956-9"
inkscape:connector-curvature="0"
sodipodi:nodetypes="cc" />
<path
style="fill:none;stroke:#000000;stroke-width:0.26458332px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1"
d="M 88.906386,76.196101 178.26877,66.880935"
id="path956-2"
inkscape:connector-curvature="0"
sodipodi:nodetypes="cc" />
<path
style="fill:none;stroke:#000000;stroke-width:0.26458332px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1"
d="M 20.619022,29.691203 21.342295,46.87499"
id="path956-0"
inkscape:connector-curvature="0"
sodipodi:nodetypes="cc" />
<g
id="g860"
transform="translate(12.428033,22.851544)">
<rect
y="18.747269"
x="68.663109"
height="12.077432"
width="48.42609"
id="rect839"
style="fill:#ff0000;fill-opacity:1;stroke:#000000;stroke-width:0.21469432;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
<flowRoot
transform="matrix(0.26458333,0,0,0.26458333,-0.80180858,-5.8799296)"
style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#000000;fill-opacity:1;stroke:none"
id="flowRoot847"
xml:space="preserve"><flowRegion
id="flowRegion849"><rect
y="86.103172"
x="275.77164"
height="55.558392"
width="163.64471"
id="rect851" /></flowRegion><flowPara
style="font-size:17.33333397px"
id="flowPara853">Vending Machines</flowPara></flowRoot> </g>
<g
id="g875"
transform="translate(-26.860587,-37.150464)">
<rect
y="84.174919"
x="34.399231"
height="8.7096987"
width="27.685835"
id="rect841"
style="fill:#00ffff;fill-opacity:1;stroke:#000000;stroke-width:0.1079253;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
<flowRoot
transform="matrix(0.26458333,0,0,0.26458333,-0.80180858,-12.027129)"
style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#000000;fill-opacity:1;stroke:none"
id="flowRoot862"
xml:space="preserve"><flowRegion
id="flowRegion864"><rect
y="349.75299"
x="144.45181"
height="48.487339"
width="91.923897"
id="rect866" /></flowRegion><flowPara
style="font-size:17.33333397px"
id="flowPara868">Locations</flowPara></flowRoot> </g>
<g
id="g914"
transform="translate(43.164029,-33.94323)">
<rect
y="95.394814"
x="135.15442"
height="9.5223551"
width="26.093069"
id="rect843"
style="fill:#00ccff;fill-opacity:1;stroke:#000000;stroke-width:0.0970764;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
<flowRoot
transform="matrix(0.26458333,0,0,0.26458333,0.53453906,-11.49259)"
style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#000000;fill-opacity:1;stroke:none"
id="flowRoot877"
xml:space="preserve"><flowRegion
id="flowRegion879"><rect
y="390.15909"
x="524.26917"
height="54.548248"
width="220.21323"
id="rect881" /></flowRegion><flowPara
style="font-size:17.33333397px"
id="flowPara883">Players</flowPara></flowRoot> </g>
<g
id="g945"
transform="translate(31.671439,-93.009795)">
<rect
y="164.60806"
x="35.720619"
height="8.7396231"
width="21.568562"
id="rect845"
style="fill:#00ff00;fill-opacity:1;stroke:#000000;stroke-width:0.07800145;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
<flowRoot
transform="matrix(0.26458333,0,0,0.26458333,-98.724537,57.6412)"
style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#000000;fill-opacity:1;stroke:none"
id="flowRoot877-1"
xml:space="preserve"><flowRegion
id="flowRegion879-6"><rect
y="390.15909"
x="524.26917"
height="54.548248"
width="220.21323"
id="rect881-4" /></flowRegion><flowPara
style="font-size:17.33333397px"
id="flowPara883-1">Deals</flowPara></flowRoot> </g>
<g
id="g954"
transform="translate(65.748303,-34.611403)">
<rect
y="156.89734"
x="55.004101"
height="10.530357"
width="31.110111"
id="rect947"
style="fill:#8dd35f;fill-opacity:1;stroke:#000000;stroke-width:0.15815331;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
<flowRoot
transform="matrix(0.26458333,0,0,0.26458333,-81.306054,50.584242)"
style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#000000;fill-opacity:1;stroke:none"
id="flowRoot877-1-2"
xml:space="preserve"><flowRegion
id="flowRegion879-6-4"><rect
y="390.15909"
x="524.26917"
height="54.548248"
width="220.21323"
id="rect881-4-1" /></flowRegion><flowPara
style="font-size:17.33333397px"
id="flowPara883-1-4">Placements</flowPara></flowRoot> </g>
<flowRoot
xml:space="preserve"
id="flowRoot996"
style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#000000;fill-opacity:1;stroke:none"
transform="matrix(0.26458333,0,0,0.26458333,-16.437076,-4.1426777)"><flowRegion
id="flowRegion998"><rect
id="rect1000"
width="210.11172"
height="52.527935"
x="623.2641"
y="182.57274" /></flowRegion><flowPara
id="flowPara1002"
style="font-size:17.33333397px">Owned by</flowPara></flowRoot> <g
id="g1044"
transform="translate(0.80180858,1.3363476)">
<rect
y="19.619349"
x="10.869535"
height="8.7296562"
width="18.351358"
id="rect841-5"
style="fill:#80b3ff;fill-opacity:1;stroke:#000000;stroke-width:0.08796817;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
<flowRoot
transform="matrix(0.26458333,0,0,0.26458333,-24.321526,-76.57272)"
style="font-style:normal;font-weight:normal;font-size:40px;line-height:1.25;font-family:sans-serif;letter-spacing:0px;word-spacing:0px;fill:#000000;fill-opacity:1;stroke:none"
id="flowRoot862-3"
xml:space="preserve"><flowRegion
id="flowRegion864-9"><rect
y="349.75299"
x="144.45181"
height="48.487339"
width="91.923897"
id="rect866-6" /></flowRegion><flowPara
style="font-size:17.33333397px"
id="flowPara868-9">Cities</flowPara></flowRoot> </g>
</g>
</svg>

After

Width:  |  Height:  |  Size: 9.6 KiB

12
package.json Normal file
View File

@ -0,0 +1,12 @@
{
"name": "vending-tycoon-cli",
"version": "1.0.0",
"main": "index.js",
"author": "Mason Payne <gomas.bmw@gmail.com>",
"license": "UNLICENCED",
"private": true,
"dependencies": {
"crypto-random-string": "^3.2.0",
"unique-names-generator": "^4.2.0"
}
}

317
src/mockProcess.ts Normal file
View File

@ -0,0 +1,317 @@
import mockData from './mockdata';
import * as cryptoRandomString from 'crypto-random-string';
import { uniqueNamesGenerator, Config, adjectives, names, colors } from 'unique-names-generator';
// function calculateBusinessDay(player: any) {
// let lineItems = [];
// for (let i in player.vendingMachines) {
// let machine = player.vendingMachines[i];
//
// }
// }
function handleEvent(eventLog: EventLog) {
if (eventLog.eventName === 'bought') {
let vendingMachine = findVendingMachine(eventLog.data.offer);
let selection = vendingMachine.selections.find((sel) => {
return sel.id === eventLog.data.selection;
});
selection.current = selection.current - 1;
}
}
function findVendingMachine(id: string) {
for (let i in mockData.players) {
let player = mockData.players[i];
for (let j in player.vendingMachines) {
if (player.vendingMachines[i].id === id) {
return player.vendingMachines[i];
}
}
}
}
function personPassesByVendingMachine(person: Person, machine): EventLog {
let isThursty: boolean = Math.random() < person.thirst;
let availableSelections = machine.selections.filter((selection) => {
return !!selection.current;
});
if (isThursty) {
if (!availableSelections.length) {
return wontBuy('NoneAvailable');
}
// pick a drink
let acceptableOptions = availableSelections.filter((option) => {
return person.preferences.indexOf(option.drink) !== -1;
});
if (!acceptableOptions.length) {
return wontBuy('NoPreferredDrinks');
}
let options = acceptableOptions.filter((selection) => {
return selection.price <= person.highestPrice;
});
if (!options.length) {
return wontBuy('HighPrice');
}
let picked = pick(acceptableOptions);
let selectionTypeData = getSelectionType(picked.type);
return {
eventName: 'bought',
data: {
reason: 'Thirsty',
amount: picked.price,
drink: picked.drink,
selection: picked.id,
selectionType: picked.type,
oz: selectionTypeData ? selectionTypeData.oz : 12,
person: person.id,
offer: machine.id,
}
};
} else {
return wontBuy('NotThirsty');
}
function wontBuy(reason): EventLog {
return {
eventName: 'wontBuy',
data: {
reason: reason,
amount: 0,
drink: 'none',
selection: 'none',
selectionType: 'none',
oz: 0,
person: person.id,
offer: machine.id,
}
};
}
}
function generatePerson(location: Location): Person {
let money = Math.random() * 100;
return {
id: genId('person'),
name: genPersonName(),
preferences: genPreferences(),
highestPrice: (money / 20),
money: money,
thirst: clamp(location.dailyTraffic.thirstBonus + (Math.random() / 10))
}
}
function genId(prefix: string): string {
return `${prefix}-${cryptoRandomString({length: 6, type: 'distinguishable'})}`;
}
function genPersonName(): string {
const config: Config = {
dictionaries: [names, names],
length: 2,
separator: ' '
};
return uniqueNamesGenerator(config);
}
function genDrinkName(): string {
const config: Config = {
dictionaries: [adjectives, colors],
length: 2,
separator: ' ',
style: 'capital'
};
return uniqueNamesGenerator(config);
}
function genStoreName(): string {
const stores: string[] = [
'shop',
'reseller',
'department Store',
'chain Store',
'emporium',
'supermarket',
'hypermarket',
'superstore',
'mart',
'shed',
'big Box',
'convenience Store',
'corner Store',
'food Store',
'market',
'food Mart',
'bodega',
];
const config: Config = {
dictionaries: [names, stores],
length: 2,
separator: '\'s ',
style: 'capital'
};
return uniqueNamesGenerator(config);
}
function genCityName(): string {
const places: string[] = [
'town',
'city',
'township',
'borough',
'seat',
'metropolis',
'burg',
'municipality',
'hamlet',
];
const config: Config = {
dictionaries: [Math.random() > 0.5 ? colors : names, places],
length: 2,
separator: ' ',
style: 'capital'
};
return uniqueNamesGenerator(config);
}
function clamp(value: number, max: number = 1, min: number = 0) {
if (value > max) {
return max;
}
if (value < min) {
return min;
}
return value;
}
function genPreferences(): string[] {
const availableDrinks: string[] = mockData.drinks;
let prefCount = Math.floor(Math.random() * availableDrinks.length);
prefCount = prefCount === 0 ? prefCount + 1 : prefCount; // have at least one preference
let preferences: string[] = [];
while (preferences.length < prefCount) {
let newPref: Drink = pick(availableDrinks);
if (preferences.indexOf(newPref.id) === -1) {
preferences.push(newPref.id);
}
}
return preferences;
}
function getSelectionType(typeName: string) {
return mockData.selectionTypes.find((type) => {
return type.type === typeName;
});
}
function randomNumber(min: number, max: number): number {
return Math.random() * (max - min) + min;
}
function pick(arr: any[]) {
return arr[Math.floor(Math.random() * arr.length)];
}
function getStatsFromEvents(events: EventLog[]) {
let purchaseCount: number = 0;
let offerEventCount: number = 0;
let amountEarned: number = 0;
let ozSold: number = 0;
let popularity: {[drink: string]: number} = {};
let eventReasons: {[reason: string]: number} = {};
for (let i in events) {
let eventLog: EventLog = events[i];
if (!eventReasons[eventLog.data.reason]) {
eventReasons[eventLog.data.reason] = 0;
}
eventReasons[eventLog.data.reason] = eventReasons[eventLog.data.reason] + 1;
if (eventLog.eventName === 'bought' || eventLog.eventName === 'wontBuy') {
offerEventCount = offerEventCount + 1;
if (eventLog.eventName === 'bought') {
purchaseCount = purchaseCount + 1;
if (!popularity[eventLog.data.drink]) {
popularity[eventLog.data.drink] = 0;
}
popularity[eventLog.data.drink] = popularity[eventLog.data.drink] + 1;
amountEarned = amountEarned + eventLog.data.amount;
ozSold = ozSold + eventLog.data.oz;
}
}
}
console.log({
purchaseCount,
purchaseRate: purchaseCount / offerEventCount,
offerEventCount,
amountEarned,
ozSold,
popularity,
eventReasons
});
}
interface Drink {
id: string;
name: string;
costPerOz: number;
}
interface DailyTraffic {
max: number;
min: number;
thirstBonus: number;
}
interface Placement {
id: string;
name: string;
traffic: number;
}
interface Location {
id: string;
name: string;
city: string;
dailyTraffic: DailyTraffic;
placements: Placement[];
}
interface Person {
id: string;
name: string;
preferences: string[];
highestPrice: number;
money: number;
thirst: number;
}
interface EventLog {
eventName: string;
data: any;
}
interface PersonEvent {
reason: string;
amount: number;
drink: string;
selection: string;
selectionType: string;
oz: number;
person: string; // person involved in the event
offer: string; // id of deal or vending machine
}
let events: EventLog[] = [];
let traffic: number = randomNumber(mockData.locations[0].dailyTraffic.min, mockData.locations[0].dailyTraffic.max)
for (let i = 0; i < traffic; i++) {
// console.log(generatePerson(mockData.locations[0]));
let curEvent: EventLog = personPassesByVendingMachine(generatePerson(mockData.locations[0]), mockData.players[0].vendingMachines[0])
// console.log(genDrinkName());
// console.log(genStoreName());
console.log(genCityName());
events.push(curEvent);
handleEvent(curEvent);
}
getStatsFromEvents(events);

219
src/mockdata.ts Normal file
View File

@ -0,0 +1,219 @@
let dataDefinition: any = {
players: [
{
id: 'player-01',
name: 'payne8',
balance: 4000.00,
vendingMachines: [
{
id: 'vend-01',
name: "First One!",
location: 'loc-01',
placement: 'plac-01',
compatibleSelectionTypes: ['can', 'bottle'],
selections: [
{
id: 'sel-01',
name: 'Diet Mtn. Dew',
drink: 'drink-01',
max: 56,
current: 24,
price: 3.00,
type: 'can'
},
{
id: 'sel-02',
name: 'Mtn. Dew',
drink: 'drink-02',
max: 56,
current: 24,
price: 2.00,
type: 'can'
},
{
id: 'sel-03',
name: 'Dr. Pepper',
drink: 'drink-03',
max: 56,
current: 24,
price: 1.50,
type: 'can'
},
{
id: 'sel-04',
name: 'Sprite',
drink: 'drink-04',
max: 56,
current: 24,
price: 0.75,
type: 'can'
},
{
id: 'sel-05',
name: 'Coke',
drink: 'drink-05',
max: 56,
current: 24,
price: 1.00,
type: 'can'
},
{
id: 'sel-06',
name: 'Diet Coke',
drink: 'drink-06',
max: 56,
current: 24,
price: 1.50,
type: 'can'
},
{
id: 'sel-07',
name: 'Orange Soda',
drink: 'drink-07',
max: 56,
current: 24,
price: 2.00,
type: 'can'
},
{
id: 'sel-08',
name: 'Grape Soda',
drink: 'drink-08',
max: 56,
current: 24,
price: 0.50,
type: 'can'
},
],
agreement: 0.0, // revenue share
}
],
deals: [
{
id: 'deal-01',
location: 'loc-01',
selections: [
{
id: 'sel-09',
name: 'Diet Mtn. Dew - Dozen Cans',
drink: 'drink-01',
max: 56,
current: 24,
price: 7.00,
type: 'dozen-box'
},
],
agreement: 0.4, // revenue share
},
]
}
],
selectionTypes: [
{
type: 'can',
oz: 12,
},
{
type: 'dozen-box',
os: 144
}
],
drinks: [
{
id: 'drink-01',
name: 'Diet Mtn. Dew',
costPerOz: 0.005,
},
{
id: 'drink-02',
name: 'Mtn. Dew',
costPerOz: 0.005,
},
{
id: 'drink-03',
name: 'Dr. Pepper',
costPerOz: 0.005,
},
{
id: 'drink-04',
name: 'Sprite',
costPerOz: 0.005,
},
{
id: 'drink-05',
name: 'Coke',
costPerOz: 0.005,
},
{
id: 'drink-06',
name: 'Diet Coke',
costPerOz: 0.005,
},
{
id: 'drink-07',
name: 'Orange Soda',
costPerOz: 0.005,
},
{
id: 'drink-08',
name: 'Grape Soda',
costPerOz: 0.005,
},
],
locations: [
{
id: 'loc-01',
name: 'General Store',
city: 'city-01',
dailyTraffic: {
max: 2000,
min: 400,
thirstBonus: 0.05,
},
deals: {
count: 2,
traffic: 0.4,
list: [
'deal-01'
]
},
placements: [
{
id: 'plac-01',
traffic: 0.5,
},
{
id: 'plac-02',
traffic: 0.6,
},
]
}
],
cities: [
{
id: 'city-01',
name: 'Hometown',
population: 15000
}
],
people: [
{
id: 'person-01',
name: 'Steve Klemm',
preferences: [
'drink-01',
'drink-04',
'drink-08',
],
highestPrice: 2.00,
wallet: {
cc: 500.00,
cash: 10.00,
coins: 1.25,
},
thirst: 0.8,
}
]
};
export default dataDefinition;

28
tsconfig.json Normal file
View File

@ -0,0 +1,28 @@
{
"compilerOptions": {
"target": "es6",
"module": "commonjs",
"moduleResolution": "node",
"isolatedModules": false,
"jsx": "react",
"experimentalDecorators": true,
"emitDecoratorMetadata": true,
"declaration": false,
"noImplicitAny": false,
"noImplicitUseStrict": false,
"removeComments": true,
"noLib": false,
"preserveConstEnums": true,
"suppressImplicitAnyIndexErrors": true,
"outDir": "./build",
"rootDir": "./src/",
"sourceMap": false
},
"compileOnSave": true,
"buildOnSave": true,
"exclue": [
"node_modules",
"typings/browser",
"typings/browser.d.ts"
]
}

20
yarn.lock Normal file
View File

@ -0,0 +1,20 @@
# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
# yarn lockfile v1
crypto-random-string@^3.2.0:
version "3.2.0"
resolved "https://registry.yarnpkg.com/crypto-random-string/-/crypto-random-string-3.2.0.tgz#d513ef0c2ac6ff7cad5769de585d9bf2ad5a2b4d"
integrity sha512-8vPu5bsKaq2uKRy3OL7h1Oo7RayAWB8sYexLKAqvCXVib8SxgbmoF1IN4QMKjBv8uI8mp5gPPMbiRah25GMrVQ==
dependencies:
type-fest "^0.8.1"
type-fest@^0.8.1:
version "0.8.1"
resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.8.1.tgz#09e249ebde851d3b1e48d27c105444667f17b83d"
integrity sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==
unique-names-generator@^4.2.0:
version "4.2.0"
resolved "https://registry.yarnpkg.com/unique-names-generator/-/unique-names-generator-4.2.0.tgz#e1845155becb4f1d133ec12fc7efc32d0cb733b1"
integrity sha512-MaNgUy/U7jeIMr5n8tdYXFi5IJ9DVo3nc90Z0QDnbYmO0F/H3Swjb8H/bkHRQYBCNveBNU6rq90metcmxVdO6Q==