2020-08-11 11:55:43 +00:00
|
|
|
class Buyer {
|
2020-08-12 08:26:20 +00:00
|
|
|
constructor(market) {
|
|
|
|
this.market = market;
|
|
|
|
}
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
/**
|
|
|
|
* Build a list of the sellers
|
|
|
|
* @param product
|
|
|
|
* @returns {*}
|
|
|
|
*/
|
|
|
|
buildSellerList(product) {
|
|
|
|
const sellers = [];
|
|
|
|
this.market.sellers.forEach((seller, i) => {
|
|
|
|
if (seller.inventory.hasOwnProperty(product)) {
|
|
|
|
const price = seller.quote(product);
|
|
|
|
sellers.push({ 'id': seller.id, 'price': price, 'index': i, 'quantity': seller.inventory[product].quantity });
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return sellers;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-08-11 23:29:04 +00:00
|
|
|
* Build an order preference table for particular product
|
|
|
|
* @param product
|
|
|
|
*/
|
2020-08-12 08:26:20 +00:00
|
|
|
buildOrderPrefBestPrice(product) {
|
|
|
|
const sellers = this.buildSellerList(product);
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
// sort by best price
|
|
|
|
sellers.sort((a, b) => {
|
|
|
|
return a.price - b.price;
|
|
|
|
});
|
2020-08-11 11:55:43 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
return sellers;
|
|
|
|
}
|
2020-08-11 11:55:43 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
/**
|
|
|
|
* Sort the list based on Delivery delay
|
|
|
|
* @param product
|
|
|
|
* @returns {*}
|
|
|
|
*/
|
|
|
|
buildOrderPrefFastFill(product) {
|
|
|
|
const sellers = this.buildSellerList(product);
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
// sort by best price
|
|
|
|
sellers.sort((a, b) => {
|
|
|
|
return a.deliveryWait - b.deliveryWait;
|
|
|
|
});
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
return sellers;
|
|
|
|
}
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
/**
|
|
|
|
* Sort the list based on item quantity
|
|
|
|
* @param product
|
|
|
|
* @returns {*}
|
|
|
|
*/
|
|
|
|
buildOrderPrefMostFill(product) {
|
|
|
|
const sellers = this.buildSellerList(product);
|
2020-08-11 11:55:43 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
// sort by best price
|
|
|
|
sellers.sort((a, b) => {
|
|
|
|
return b.quantity - a.quantity;
|
|
|
|
});
|
2020-08-11 11:55:43 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
return sellers;
|
|
|
|
}
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
/**
|
2020-08-11 23:29:04 +00:00
|
|
|
* This method should get the best price for a given product
|
|
|
|
* across all sellers
|
|
|
|
*/
|
2020-08-12 08:26:20 +00:00
|
|
|
getBestPrice(product) {
|
|
|
|
let lowestPrice = null;
|
|
|
|
this.market.sellers.forEach(seller => {
|
|
|
|
if (seller.inventory.hasOwnProperty(product)) {
|
|
|
|
const price = seller.quote(product);
|
|
|
|
if (lowestPrice === null || lowestPrice > price) lowestPrice = price;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return lowestPrice || 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Do the actual purchases based on preference list
|
|
|
|
* @param preference
|
|
|
|
* @param product
|
|
|
|
* @param quantity
|
|
|
|
* @returns {number}
|
|
|
|
*/
|
|
|
|
doPurchases(preference, product, quantity) {
|
|
|
|
const sellerPreference = [...preference];
|
|
|
|
let total = 0;
|
|
|
|
let wantedQuantity = quantity;
|
|
|
|
const receipt = [];
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
while (sellerPreference.length > 0 && wantedQuantity > 0) {
|
|
|
|
const seller = sellerPreference.shift();
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
const r = this.market.sellers[seller.index].sell(product, wantedQuantity);
|
|
|
|
wantedQuantity = (wantedQuantity - r.boughtQuantity) < 0 ? 0 : (wantedQuantity - r.boughtQuantity);
|
|
|
|
|
|
|
|
receipt.push(r);
|
2020-08-11 23:29:04 +00:00
|
|
|
}
|
2020-08-11 11:55:43 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
if (receipt.length > 1)
|
|
|
|
total = receipt.reduce((a, cv) => {
|
|
|
|
return (typeof a === 'number' ? a : a.cost) + cv.cost;
|
|
|
|
});
|
|
|
|
else if (receipt.length === 1) total = receipt[0].cost;
|
2020-08-11 11:55:43 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
return total;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-08-11 23:29:04 +00:00
|
|
|
* This method should optimise price when filling an order
|
|
|
|
* if the quantity is greater than any single seller can accomodate
|
|
|
|
* then the next cheapest seller should be used.
|
|
|
|
*/
|
2020-08-12 08:26:20 +00:00
|
|
|
fillWithBestPrices(product, quantity) {
|
|
|
|
const sellerPreference = this.buildOrderPrefBestPrice(product);
|
2020-08-11 11:55:43 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
return this.doPurchases(sellerPreference, product, quantity);
|
|
|
|
}
|
2020-08-11 20:12:17 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
/**
|
2020-08-11 23:29:04 +00:00
|
|
|
* This method should optimise for sellers with the largest inventory when filling an order
|
|
|
|
* if the quantity is greater than any single seller can accomodate
|
|
|
|
* then the next largest seller should be used.
|
|
|
|
* if multiple sellers have the same amount of inventory
|
|
|
|
* you should use the cheaper of the two.
|
|
|
|
*/
|
2020-08-12 08:26:20 +00:00
|
|
|
fillWithLargestSellers(product, quantity) {
|
|
|
|
const sellerPreference = this.buildOrderPrefMostFill(product);
|
2020-08-11 11:55:43 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
return this.doPurchases(sellerPreference, product, quantity);
|
|
|
|
}
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
/**
|
2020-08-11 23:29:04 +00:00
|
|
|
* This fulfils orders based on time to deliver
|
|
|
|
* @param product
|
|
|
|
* @param quantity
|
|
|
|
* @returns {number}
|
|
|
|
*/
|
2020-08-12 08:26:20 +00:00
|
|
|
quicklyFill(product, quantity) {
|
|
|
|
const sellerPreference = this.buildOrderPrefFastFill(product);
|
2020-08-11 23:29:04 +00:00
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
return this.doPurchases(sellerPreference, product, quantity);
|
|
|
|
}
|
2020-08-11 11:55:43 +00:00
|
|
|
}
|
|
|
|
|
2020-08-12 08:26:20 +00:00
|
|
|
module.exports = { Buyer };
|