Files
pyfa/service/market.py
2010-10-26 00:01:31 +02:00

280 lines
8.7 KiB
Python

#===============================================================================
# Copyright (C) 2010 Diego Duclos
#
# This file is part of pyfa.
#
# pyfa is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# pyfa is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with pyfa. If not, see <http://www.gnu.org/licenses/>.
#===============================================================================
import eos.db
import eos.types
import eos.saveddata
import wx
import threading
from sqlalchemy.orm.exc import NoResultFound
import Queue
import traceback
class PriceWorkerThread(threading.Thread):
def run(self):
self.queue = Queue.Queue()
self.processUpdates()
def processUpdates(self):
queue = self.queue
while True:
try:
# Grab our data and rerelease the lock
callback, requests = queue.get()
# Grab prices, this is the time-consuming part
if len(requests) > 0:
eos.types.Price.fetchPrices(*requests)
wx.CallAfter(callback)
except:
pass
finally:
try:
queue.task_done()
except:
pass
def trigger(self, prices, callbacks):
self.queue.put((callbacks, prices))
class SearchWorkerThread(threading.Thread):
def run(self):
self.cv = threading.Condition()
self.searchRequest = None
self.processSearches()
def processSearches(self):
cv = self.cv
while True:
cv.acquire()
while self.searchRequest is None:
cv.wait()
request, callback = self.searchRequest
self.searchRequest = None
cv.release()
filter = (eos.types.Category.name.in_(Market.SEARCH_CATEGORIES), eos.types.Item.published == True)
results = eos.db.searchItems(request, where=filter,
join=(eos.types.Item.group, eos.types.Group.category),
eager=("icon", "group.category"))
items = []
for item in results:
if item.category.name in Market.SEARCH_CATEGORIES:
items.append((item.ID, item.name, item.group.name, item.metaGroup.ID if item.metaGroup else 1, item.icon.iconFile if item.icon else ""))
wx.CallAfter(callback, items)
def scheduleSearch(self, text, callback):
self.cv.acquire()
self.searchRequest = (text, callback)
self.cv.notify()
self.cv.release()
class Market():
instance = None
FORCED_SHIPS = ("Freki", "Mimir", "Utu", "Adrestia", "Ibis", "Impairor", "Velator", "Reaper")
FORCED_GROUPS = ("Rookie ship",)
META_MAP = {"normal": (1, 2, 14),
"faction": (4, 3),
"complex": (6,),
"officer": (5,)}
SEARCH_CATEGORIES = ("Drone", "Module", "Subsystem", "Charge", "Implant")
@classmethod
def getInstance(cls):
if cls.instance == None:
cls.instance = Market()
return cls.instance
def __init__(self):
self.activeMetas = set()
self.priceCache = {}
self.priceWorkerThread = PriceWorkerThread()
self.priceWorkerThread.daemon = True
self.priceWorkerThread.start()
self.searchWorkerThread = SearchWorkerThread()
self.searchWorkerThread.daemon = True
self.searchWorkerThread.start()
def getChildren(self, id):
"""
Get the children of the group or marketGroup with the passed id.
Returns a list, where each element is a tuple containing:
the id, the name, the icon, wether the group has more children.
"""
group = eos.db.getMarketGroup(id, eager="icon")
children = []
for child in group.children:
children.append((child.ID, child.name, self.figureIcon(child), not child.hasTypes))
return children
def getShipRoot(self):
cat = eos.db.getCategory(6)
root = []
for grp in cat.groups:
if grp.published or grp.name in self.FORCED_GROUPS:
root.append((grp.ID, grp.name))
return root
def getShipList(self, id):
ships = []
grp = eos.db.getGroup(id, eager=("items", "items.marketGroup", "items.attributes"))
for item in grp.items:
if item.published or item.name in self.FORCED_SHIPS:
ships.append((item.ID, item.name, item.race))
return ships
def searchShips(self, name):
results = eos.db.searchItems(name)
ships = []
for item in results:
if item.category.name == "Ship" and (item.published or item.name in self.FORCED_SHIPS):
ships.append((item.ID, item.name, item.race))
return ships
def searchItems(self, name, callback):
self.searchWorkerThread.scheduleSearch(name, callback)
def searchFits(self, name):
results = eos.db.searchFits(name)
fits = []
for fit in results:
fits.append((fit.ID, fit.name, fit.ship.item.name))
return fits
def getImplantTree(self):
return self.getChildren(27)
def getItem(self, itemId):
return eos.db.getItem(itemId)
def getGroup(self, groupId):
return eos.db.getGroup(groupId)
MARKET_GROUPS = (9, #Modules
1111, #Rigs
157, #Drones
11, #Ammo
1112, #Subsystems
24) #Implants & Boosters
def getMarketRoot(self):
"""
Get the root of the market tree.
Returns a list, where each element is a tuple containing:
the ID, the name and the icon of the group
"""
root = []
for id in self.MARKET_GROUPS:
mg = eos.db.getMarketGroup(id, eager="icon")
root.append((id, mg.name, self.figureIcon(mg)))
return root
def figureIcon(self, mg):
if mg.icon:
return mg.icon.iconFile
else:
if mg.hasTypes and len(mg.items) > 0:
item = mg.items[0]
return item.icon.iconFile if item.icon else ""
elif len(mg.children) > 0:
return self.figureIcon(mg.children[0])
else:
return ""
def activateMetaGroup(self, name):
for meta in self.META_MAP[name]:
self.activeMetas.add(meta)
def disableMetaGroup(self, name):
for meta in self.META_MAP[name]:
if meta in self.activeMetas:
self.activeMetas.remove(meta)
def isMetaIdActive(self, meta):
return meta in self.activeMetas
def getMetaName(self, metaId):
for name, ids in self.META_MAP.items():
for id in ids:
if metaId == id:
return name
def getVariations(self, marketGroupId):
if len(self.activeMetas) == 0:
return tuple()
mg = eos.db.getMarketGroup(marketGroupId)
l = []
done = set()
populatedMetas = set()
for item in mg.items:
populatedMetas.add(1)
if 1 in self.activeMetas:
if item not in done:
done.add(item)
l.append((item.ID, item.name, item.icon.iconFile if item.icon else ""))
vars = eos.db.getVariations(item, eager=("icon", "metaGroup"))
for var in vars:
populatedMetas.add(var.metaGroup.ID)
if var not in done and var.metaGroup.ID in self.activeMetas:
done.add(var)
l.append((var.ID, var.name, var.icon.iconFile if var.icon else ""))
return l, populatedMetas
def getPrices(self, typeIDs, callback):
requests = []
for typeID in typeIDs:
price = self.priceCache.get(typeID)
if price is None:
try:
price = eos.db.getPrice(typeID)
except NoResultFound:
price = eos.types.Price(typeID)
eos.db.saveddata_session.add(price)
self.priceCache[typeID] = price
requests.append(price)
def cb():
callback(requests)
eos.db.commit()
self.priceWorkerThread.trigger(requests, cb)