package appeng.me.cache; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.Comparator; import java.util.HashMap; import java.util.HashSet; import java.util.Iterator; import java.util.List; import java.util.Map.Entry; import java.util.Set; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Future; import java.util.concurrent.ThreadFactory; import net.minecraft.world.World; import appeng.api.config.AccessRestriction; import appeng.api.config.Actionable; import appeng.api.networking.IGrid; import appeng.api.networking.IGridHost; import appeng.api.networking.IGridNode; import appeng.api.networking.IGridStorage; import appeng.api.networking.crafting.ICraftingCPU; import appeng.api.networking.crafting.ICraftingCallback; import appeng.api.networking.crafting.ICraftingGrid; import appeng.api.networking.crafting.ICraftingJob; import appeng.api.networking.crafting.ICraftingLink; import appeng.api.networking.crafting.ICraftingMedium; import appeng.api.networking.crafting.ICraftingPatternDetails; import appeng.api.networking.crafting.ICraftingProvider; import appeng.api.networking.crafting.ICraftingProviderHelper; import appeng.api.networking.crafting.ICraftingRequester; import appeng.api.networking.crafting.ICraftingWatcher; import appeng.api.networking.crafting.ICraftingWatcherHost; import appeng.api.networking.energy.IEnergyGrid; import appeng.api.networking.events.MENetworkCraftingCpuChange; import appeng.api.networking.events.MENetworkCraftingPatternChange; import appeng.api.networking.events.MENetworkEventSubscribe; import appeng.api.networking.events.MENetworkPostCacheConstruction; import appeng.api.networking.security.BaseActionSource; import appeng.api.networking.storage.IStorageGrid; import appeng.api.storage.ICellProvider; import appeng.api.storage.IMEInventoryHandler; import appeng.api.storage.StorageChannel; import appeng.api.storage.data.IAEItemStack; import appeng.api.storage.data.IAEStack; import appeng.api.storage.data.IItemList; import appeng.crafting.CraftingJob; import appeng.crafting.CraftingLink; import appeng.crafting.CraftingLinkNexus; import appeng.crafting.CraftingWatcher; import appeng.me.cluster.implementations.CraftingCPUCluster; import appeng.me.helpers.GenericInterestManager; import appeng.me.storage.ItemWatcher; import appeng.tile.crafting.TileCraftingStorageTile; import appeng.tile.crafting.TileCraftingTile; import appeng.util.ItemSorters; import com.google.common.collect.HashMultimap; import com.google.common.collect.ImmutableCollection; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableSet; import com.google.common.collect.SetMultimap; public class CraftingGridCache implements ICraftingGrid, ICraftingProviderHelper, ICellProvider, IMEInventoryHandler { HashSet cpuClusters = new HashSet(); HashSet providers = new HashSet(); private HashMap watchers = new HashMap(); IGrid grid; IStorageGrid sg; IEnergyGrid eg; HashMap> craftingMethods = new HashMap(); HashMap> craftableItems = new HashMap(); HashSet emitableItems = new HashSet(); HashMap links = new HashMap(); boolean updateList = false; final private SetMultimap interests = HashMultimap.create(); final public GenericInterestManager interestManager = new GenericInterestManager( interests ); class ActiveCpuIterator implements Iterator { final Iterator i; CraftingCPUCluster c = null; public ActiveCpuIterator(Collection o) { i = o.iterator(); } @Override public boolean hasNext() { findNext(); return c != null; } private void findNext() { while (i.hasNext() && c == null) { c = i.next(); if ( !c.isActive() || c.isDestroyed ) c = null; } } @Override public ICraftingCPU next() { ICraftingCPU o = c; c = null; return o; } @Override public void remove() { // no.. } }; @Override public ImmutableSet getCpus() { return ImmutableSet.copyOf( new ActiveCpuIterator( cpuClusters ) ); } public CraftingGridCache(IGrid g) { grid = g; } @MENetworkEventSubscribe public void afterCacheConstruction(MENetworkPostCacheConstruction cc) { sg = grid.getCache( IStorageGrid.class ); eg = grid.getCache( IEnergyGrid.class ); sg.registerCellProvider( this ); } public void addLink(CraftingLink l) { if ( l.isStandalone() ) return; CraftingLinkNexus n = links.get( l.getCraftingID() ); if ( n == null ) links.put( l.getCraftingID(), n = new CraftingLinkNexus( l.getCraftingID() ) ); l.setNextus( n ); } @Override public void onUpdateTick() { if ( updateList ) { updateList = false; updateCPUClusters(); } Iterator i = links.values().iterator(); while (i.hasNext()) { if ( i.next().isDead( grid, this ) ) i.remove(); } for (CraftingCPUCluster cpu : cpuClusters) cpu.updateCraftingLogic( grid, eg, this ); } @MENetworkEventSubscribe public void updateCPUClusters(MENetworkCraftingCpuChange c) { updateList = true; } @MENetworkEventSubscribe public void updateCPUClusters(MENetworkCraftingPatternChange c) { updatePatterns(); } @Override public void removeNode(IGridNode gridNode, IGridHost machine) { if ( machine instanceof ICraftingWatcherHost ) { ICraftingWatcher myWatcher = watchers.get( machine ); if ( myWatcher != null ) { myWatcher.clear(); watchers.remove( machine ); } } if ( machine instanceof ICraftingRequester ) { Iterator nex = links.values().iterator(); while (nex.hasNext()) { CraftingLinkNexus n = nex.next(); if ( n.isMachine( machine ) ) n.removeNode(); } } if ( machine instanceof TileCraftingTile ) updateList = true; if ( machine instanceof ICraftingProvider ) { providers.remove( machine ); updatePatterns(); } } @Override public void addNode(IGridNode gridNode, IGridHost machine) { if ( machine instanceof ICraftingWatcherHost ) { ICraftingWatcherHost swh = (ICraftingWatcherHost) machine; CraftingWatcher iw = new CraftingWatcher( this, (ICraftingWatcherHost) swh ); watchers.put( gridNode, iw ); swh.updateWatcher( iw ); } if ( machine instanceof ICraftingRequester ) { for (ICraftingLink l : ((ICraftingRequester) machine).getRequestedJobs()) { if ( l instanceof CraftingLink ) addLink( (CraftingLink) l ); } } if ( machine instanceof TileCraftingTile ) updateList = true; if ( machine instanceof ICraftingProvider ) { providers.add( (ICraftingProvider) machine ); updatePatterns(); } } private void updateCPUClusters() { cpuClusters.clear(); for (IGridNode cst : grid.getMachines( TileCraftingStorageTile.class )) { TileCraftingStorageTile tile = (TileCraftingStorageTile) cst.getMachine(); CraftingCPUCluster clust = (CraftingCPUCluster) tile.getCluster(); if ( clust != null ) { cpuClusters.add( clust ); if ( clust.myLastLink != null ) addLink( (CraftingLink) clust.myLastLink ); } } } @Override public void addCraftingOption(ICraftingMedium medium, ICraftingPatternDetails api) { List details = craftingMethods.get( api ); if ( details == null ) { details = new ArrayList(); details.add( medium ); craftingMethods.put( api, details ); } else details.add( medium ); } private void updatePatterns() { HashMap> oldItems = craftableItems; // erase list. craftingMethods.clear(); craftableItems = new HashMap(); emitableItems.clear(); // update the stuff that was in the list... sg.postAlterationOfStoredItems( StorageChannel.ITEMS, oldItems.keySet(), new BaseActionSource() ); // re-create list.. for (ICraftingProvider cp : providers) cp.provideCrafting( this ); HashMap> tmpCraft = new HashMap(); // new craftables! for (ICraftingPatternDetails details : craftingMethods.keySet()) { for (IAEItemStack out : details.getOutputs()) { out = out.copy(); out.reset(); out.setCraftable( true ); Set methods = tmpCraft.get( out ); if ( methods == null ) tmpCraft.put( out, methods = new HashSet() ); methods.add( details ); } } // make them immutable for (Entry> e : tmpCraft.entrySet()) craftableItems.put( e.getKey(), ImmutableList.copyOf( e.getValue() ) ); sg.postAlterationOfStoredItems( StorageChannel.ITEMS, craftableItems.keySet(), new BaseActionSource() ); } @Override public void onSplit(IGridStorage destinationStorage) { // nothing! } @Override public void onJoin(IGridStorage sourceStorage) { // nothing! } @Override public void populateGridStorage(IGridStorage destinationStorage) { // nothing! } @Override public List getCellArray(StorageChannel channel) { ArrayList list = new ArrayList( 1 ); if ( channel == StorageChannel.ITEMS ) list.add( this ); return list; } @Override public int getPriority() { return Integer.MAX_VALUE; } @Override public IAEStack extractItems(IAEStack request, Actionable mode, BaseActionSource src) { return null; } @Override public IItemList getAvailableItems(IItemList out) { // add craftable items! for (IAEItemStack st : craftableItems.keySet()) out.addCrafting( st ); for (IAEItemStack st : emitableItems) out.addCrafting( st ); return out; } @Override public StorageChannel getChannel() { return StorageChannel.ITEMS; } @Override public AccessRestriction getAccess() { return AccessRestriction.WRITE; } @Override public boolean isPrioritized(IAEStack input) { return true; } @Override public IAEStack injectItems(IAEStack input, Actionable type, BaseActionSource src) { for (CraftingCPUCluster cpu : cpuClusters) input = cpu.injectItems( input, type, src ); return input; } @Override public boolean canAccept(IAEStack input) { for (CraftingCPUCluster cpu : cpuClusters) if ( cpu.canAccept( (IAEItemStack) input ) ) return true; return false; } @Override public ICraftingLink submitJob(ICraftingJob job, ICraftingRequester requestingMachine, ICraftingCPU target, final boolean prioritizePower, BaseActionSource src) { if ( job.isSimulation() ) return null; CraftingCPUCluster cpuClust = null; if ( target instanceof CraftingCPUCluster ) cpuClust = (CraftingCPUCluster) target; if ( target == null ) { List validCpusClusters = new ArrayList(); for (CraftingCPUCluster cpu : cpuClusters) { if ( cpu.isActive() && !cpu.isBusy() && cpu.getAvailableStorage() >= job.getByteTotal() ) { validCpusClusters.add( cpu ); } } Collections.sort( validCpusClusters, new Comparator() { @Override public int compare(CraftingCPUCluster o1, CraftingCPUCluster o2) { if ( prioritizePower ) { int a = ItemSorters.compareLong( o2.getCoProcessors(), o1.getCoProcessors() ); if ( a != 0 ) return a; return ItemSorters.compareLong( o2.getAvailableStorage(), o1.getAvailableStorage() ); } int a = ItemSorters.compareLong( o1.getCoProcessors(), o2.getCoProcessors() ); if ( a != 0 ) return a; return ItemSorters.compareLong( o1.getAvailableStorage(), o2.getAvailableStorage() ); } } ); if ( !validCpusClusters.isEmpty() ) cpuClust = validCpusClusters.get( 0 ); } if ( cpuClust != null ) { return cpuClust.submitJob( grid, job, src, requestingMachine ); } return null; } @Override public int getSlot() { return 0; } @Override public ImmutableCollection getCraftingFor(IAEItemStack whatToCraft, ICraftingPatternDetails details, int slotIndex, World world) { ImmutableList res = craftableItems.get( whatToCraft ); if ( res == null ) { if ( details != null && details.isCraftable() ) { for (IAEItemStack ais : craftableItems.keySet()) { if ( ais.getItem() == whatToCraft.getItem() && (!ais.getItem().getHasSubtypes() || ais.getItemDamage() == whatToCraft.getItemDamage()) ) { if ( details.isValidItemForSlot( slotIndex, ais.getItemStack(), world ) ) { return craftableItems.get( ais ); } } } } return ImmutableSet.of(); } return res; } public List getMediums(ICraftingPatternDetails key) { List o = craftingMethods.get( key ); if ( o == null ) o = ImmutableList.of(); return o; } @Override public boolean validForPass(int i) { return i == 1; } final public static ExecutorService craftingPool; static { ThreadFactory factory = new ThreadFactory() { @Override public Thread newThread(Runnable ar) { return new Thread( ar, "AE Crafting Calculator" ); } }; craftingPool = Executors.newCachedThreadPool( factory ); } @Override public Future beginCraftingJob(World world, IGrid grid, BaseActionSource actionSrc, IAEItemStack slotItem, ICraftingCallback cb) { if ( world == null || grid == null || actionSrc == null || slotItem == null ) throw new RuntimeException( "Invalid Craftinb Job Request" ); CraftingJob cj = new CraftingJob( world, grid, actionSrc, slotItem, cb ); return craftingPool.submit( cj, (ICraftingJob) cj ); } public boolean hasCpu(ICraftingCPU cpu) { return cpuClusters.contains( cpu ); } @Override public boolean isRequesting(IAEItemStack what) { for (CraftingCPUCluster c : cpuClusters) if ( c.isMaking( what ) ) return true; return false; } @Override public boolean canEmitFor(IAEItemStack what) { return emitableItems.contains( what ); } @Override public void setEmitable(IAEItemStack what) { emitableItems.add( what.copy() ); } }