/* * Copyright (c) 2018, 7u83 <7u83@mail.ru> * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * * Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ package opensesim.world; import java.util.Collections; import java.util.HashMap; import java.util.HashSet; import java.util.Map; import opensesim.util.scheduler.Event; import opensesim.util.scheduler.EventListener; import org.json.JSONObject; /** * Class to hold account data of traders * * @author 7u83 <7u83@mail.ru> */ public class Account { HashMap assets = new HashMap<>(); HashMap assets_bound = new HashMap<>(); HashMap stop_los = new HashMap<>(); public double margin_bound = 0.0; Trader owner; //public Exchange exchange = null; private RealWorld world; private Exchange exchange; public Exchange getExchange() { return exchange; } private boolean unlimited = false; public boolean isUnlimied() { return unlimited; } void setUnlimied(boolean unlimied) { this.unlimited = unlimied; } private double leverage = 0.0; public double getLeverage() { return leverage; } protected void setLeverage(double leverage) { this.leverage = leverage; } public Map getAssets() { return Collections.unmodifiableMap(assets); } public Map getAssetsAavail() { return Collections.unmodifiableMap(assets_bound); } public Trader getOwner() { return owner; } protected Account(World world) { this(world, null, null); } protected Account(World world, Exchange exchange, JSONObject cfg) { this.world = (RealWorld) world; if (exchange == null) { this.exchange = world.getDefaultExchange(); } if (cfg == null) { return; } } public Double getMargin(AbstractAsset currency) { /* Double d = this.getAssetDebt(world.getDefaultExchange(), currency); Double f = this.getFinalBalance(currency) * getLeverage() ; System.out.printf("Debth %f - Final: %f Return margin %f\n", d,f, f-d); return f-d;*/ return this.getFinalBalance(currency) * getLeverage() + this.getFinalBalance(currency) - this.getAssetDebt(world.getDefaultExchange(), currency); } synchronized void add(AssetPack pack) { assets.put(pack.asset, get(pack.asset) + pack.volume); assets_bound.put(pack.asset, getAvail(pack.asset) + pack.volume); } synchronized void sub(AssetPack pack) { assets.put(pack.asset, get(pack.asset) - pack.volume); // assets_bound.put(pack.asset, getAvail(pack.asset) - pack.volume); } public double get(AbstractAsset asset) { return assets.getOrDefault(asset, 0.0); } public double getAvail(AbstractAsset asset) { if (this.getLeverage() > 0) { Double margin = this.getMargin(world.getDefaultCurrency()); AssetPair ap = world.getAssetPair(asset, world.getDefaultCurrency()); return margin / world.getDefaultExchange().getAPI(ap).getLastQuote().price; } return 0.0; //return assets_bound.getOrDefault(asset, 0.0); } public void addAvail(AbstractAsset asset, double val) { double avail = getAvail(asset); assets_bound.put(asset, (avail + val)); } HashSet listeners = new HashSet<>(); public void addListener(EventListener l) { listeners.add(l); } public void notfiyListeners() { Event e = new Event() { }; for (EventListener l : listeners) { l.receive(e); } } public Double getFreeMargin(AbstractAsset asset) { return 0.0; } public Double getAssetDebt(Exchange ex, AbstractAsset currency) { Double result = 0.0; for (AbstractAsset a : assets.keySet()) { if (a.equals(currency)) { continue; } AssetPair pair = world.getAssetPair(a, currency); if (pair == null) { continue; } TradingEngine api = (TradingEngine) ex.getAPI(pair); Double v = get(a) * api.last_quote.price; Double sl = this.calcStopLoss(a); Double n = get(a); if (n == 0.0) { continue; } // System.out.printf("Asset: %s - %f %f %f\n", a.getSymbol(), n, v, sl * n); result = result + (v - sl * n); // System.out.printf("Result is now %f\n", result); } // System.out.printf("Return Dresult %f\n", result); return result; } /** * Determine final balance of this account, as if all assets would be sold * on exchange ex against given currency asset. * * @param ex Exchange to operate on * @param currency Currency against the assets should be sold. * @return final balance * */ public Double getFinalBalance(Exchange ex, AbstractAsset currency, boolean bound) { Double result = get(currency); for (AbstractAsset a : assets.keySet()) { Double v; if (a.equals(currency)) { continue; } AssetPair pair = world.getAssetPair(a, currency); if (pair == null) { continue; } v = get(a) + (bound ? getBound(a) : 0.0); if (v == 0.0) { continue; } TradingEngine api = (TradingEngine) ex.getAPI(pair); result = result + v * api.last_quote.price; } return result; } Double getBound(AbstractAsset asset){ return assets_bound.getOrDefault(asset, 0.0); } void addBound(AbstractAsset asset, Double vol){ assets_bound.put(asset, getBound(asset)+vol); } /** * Get the final balance as if all assets would be sold ob the default * exchange against given currency. * {@link #getFinalBalance(opensesim.world.Exchange, opensesim.world.AbstractAsset)} * * @param currency Currency for final balance * @return final balance */ public Double getFinalBalance(AbstractAsset currency) { return getFinalBalance(this.getExchange(), currency, false); } /** * Determine final balance * {@link #getFinalBalance(opensesim.world.Exchange, opensesim.world.AbstractAsset) } * * @see DoublegetFinalBalance( Exchange ex, AbstractAsset currency) * @return Balance */ public Double getFinalBalance() { return getFinalBalance(world.getDefaultCurrency()); } public Double calcStopLoss(Exchange ex, AbstractAsset asset, AbstractAsset currency) { Double e = (get(currency)); for (AbstractAsset a : assets.keySet()) { if (a.equals(asset)) { continue; } AssetPair pair = world.getAssetPair(a, currency); if (pair == null) { continue; } TradingEngine api = (TradingEngine) ex.getAPI(pair); Double v = get(a) * api.last_quote.price; e = e + v; } return -(double) e / (double) get(asset); } public Double calcStopLoss(AbstractAsset asset) { return calcStopLoss(world.getDefaultExchange(), asset, world.getDefaultAssetPair().getCurrency()); } /** * Return the world this account belongs to * * @return world */ public World getWorld() { return world; } private boolean isLeveraged() { return getLeverage() > 0.0; } boolean isCovered(AssetPair pair, double volume, double limit) { if (this.isUnlimied()) { return true; } if (!this.isLeveraged()) { if (limit == 0.0) { // an unlimited order is always considered to be // covereable. When the trade comes to execution, // the limits will be checked. return true; } if (volume < 0) { // It's a limited sell order, we have just to check // if a sufficient amount of assets is available return getAvail(pair.getAsset()) + volume > 0; } // Check if enough money is available to cover the // entiere volume to by return getAvail(pair.getCurrency()) >= limit * volume; } Double margin = this.getMargin(pair.getCurrency()); System.out.printf("Margin: %f > %f\n", margin, Math.abs(volume * limit)); return margin > 0; //Math.abs(volume * limit); } }