447 lines
18 KiB
Python
Executable File
447 lines
18 KiB
Python
Executable File
#!/usr/bin/env python
|
|
#======================================================================
|
|
# Copyright (C) 2012 Diego Duclos
|
|
#
|
|
# This file is part of eos.
|
|
#
|
|
# eos is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Lesser General Public License as
|
|
# published by the Free Software Foundation, either version 3 of
|
|
# the License, or (at your option) any later version.
|
|
#
|
|
# eos 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 Lesser General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
# License along with eos. If not, see <http://www.gnu.org/licenses/>.
|
|
#======================================================================
|
|
|
|
import functools
|
|
import os
|
|
import sys
|
|
|
|
# Add eos root path to sys.path so we can import ourselves
|
|
path = os.path.dirname(__file__)
|
|
sys.path.insert(0, os.path.realpath(os.path.join(path, '..')))
|
|
|
|
import json
|
|
import argparse
|
|
import itertools
|
|
|
|
|
|
CATEGORIES_TO_REMOVE = [
|
|
30 # Apparel
|
|
]
|
|
|
|
def main(db, json_path):
|
|
if os.path.isfile(db):
|
|
os.remove(db)
|
|
|
|
jsonPath = os.path.expanduser(json_path)
|
|
|
|
# Import eos.config first and change it
|
|
import eos.config
|
|
eos.config.gamedata_connectionstring = db
|
|
eos.config.debug = False
|
|
|
|
# Now thats done, we can import the eos modules using the config
|
|
import eos.db
|
|
import eos.gamedata
|
|
|
|
# Create the database tables
|
|
eos.db.gamedata_meta.create_all()
|
|
|
|
# Config dict
|
|
tables = {
|
|
'clonegrades': eos.gamedata.AlphaCloneSkill,
|
|
'dgmattribs': eos.gamedata.AttributeInfo,
|
|
'dgmeffects': eos.gamedata.Effect,
|
|
'dgmtypeattribs': eos.gamedata.Attribute,
|
|
'dgmtypeeffects': eos.gamedata.ItemEffect,
|
|
'dgmunits': eos.gamedata.Unit,
|
|
'evecategories': eos.gamedata.Category,
|
|
'evegroups': eos.gamedata.Group,
|
|
'invmetagroups': eos.gamedata.MetaGroup,
|
|
'invmetatypes': eos.gamedata.MetaType,
|
|
'evetypes': eos.gamedata.Item,
|
|
'phbtraits': eos.gamedata.Traits,
|
|
'phbmetadata': eos.gamedata.MetaData,
|
|
'mapbulk_marketGroups': eos.gamedata.MarketGroup,
|
|
}
|
|
|
|
fieldMapping = {
|
|
'dgmattribs': {
|
|
'displayName': 'displayName'
|
|
},
|
|
'dgmeffects': {
|
|
'displayName': 'displayName',
|
|
'description': 'description'
|
|
},
|
|
'dgmunits': {
|
|
'displayName': 'displayName'
|
|
},
|
|
#icons???
|
|
'evecategories': {
|
|
'categoryName': 'categoryName'
|
|
},
|
|
'evegroups': {
|
|
'groupName': 'groupName'
|
|
},
|
|
'invmetagroups': {
|
|
'metaGroupName': 'metaGroupName'
|
|
},
|
|
'evetypes': {
|
|
'typeName': 'typeName',
|
|
'description': 'description'
|
|
},
|
|
#phbtraits???
|
|
'mapbulk_marketGroups': {
|
|
'marketGroupName': 'marketGroupName',
|
|
'description': 'description'
|
|
}
|
|
|
|
}
|
|
|
|
rowsInValues = (
|
|
'evetypes',
|
|
'evegroups',
|
|
'evecategories'
|
|
)
|
|
|
|
def convertIcons(data):
|
|
new = []
|
|
for k, v in list(data.items()):
|
|
v['iconID'] = k
|
|
new.append(v)
|
|
return new
|
|
|
|
def convertClones(data):
|
|
newData = []
|
|
|
|
# December, 2017 - CCP decided to use only one set of skill levels for alpha clones. However, this is still
|
|
# represented in the data as a skillset per race. To ensure that all skills are the same, we store them in a way
|
|
# that we can check to make sure all races have the same skills, as well as skill levels
|
|
|
|
check = {}
|
|
|
|
for ID in data:
|
|
for skill in data[ID]['skills']:
|
|
newData.append({
|
|
'alphaCloneID': int(ID),
|
|
'alphaCloneName': 'Alpha Clone',
|
|
'typeID': skill['typeID'],
|
|
'level': skill['level']})
|
|
if ID not in check:
|
|
check[ID] = {}
|
|
check[ID][int(skill['typeID'])] = int(skill['level'])
|
|
|
|
if not functools.reduce(lambda a, b: a if a == b else False, [v for _, v in check.items()]):
|
|
raise Exception('Alpha Clones not all equal')
|
|
|
|
newData = [x for x in newData if x['alphaCloneID'] == 1]
|
|
|
|
if len(newData) == 0:
|
|
raise Exception('Alpha Clone processing failed')
|
|
|
|
return newData
|
|
|
|
def convertTraits(data):
|
|
|
|
def convertSection(sectionData):
|
|
sectionLines = []
|
|
headerText = '<b>{}</b>'.format(sectionData['header'])
|
|
sectionLines.append(headerText)
|
|
for bonusData in sectionData['bonuses']:
|
|
prefix = '{} '.format(bonusData['number']) if 'number' in bonusData else ''
|
|
bonusText = '{}{}'.format(prefix, bonusData['text'].replace('\u00B7', '\u2022 '))
|
|
sectionLines.append(bonusText)
|
|
sectionLine = '<br />\n'.join(sectionLines)
|
|
return sectionLine
|
|
|
|
newData = []
|
|
for row in data:
|
|
typeLines = []
|
|
typeId = row['typeID']
|
|
traitData = row['traits']
|
|
for skillData in sorted(traitData.get('skills', ()), key=lambda i: i['header']):
|
|
typeLines.append(convertSection(skillData))
|
|
if 'role' in traitData:
|
|
typeLines.append(convertSection(traitData['role']))
|
|
if 'misc' in traitData:
|
|
typeLines.append(convertSection(traitData['misc']))
|
|
traitLine = '<br />\n<br />\n'.join(typeLines)
|
|
newRow = {'typeID': typeId, 'traitText': traitLine}
|
|
newData.append(newRow)
|
|
return newData
|
|
|
|
def fillReplacements(tables):
|
|
|
|
def compareAttrs(attrs1, attrs2, attrHig):
|
|
"""
|
|
Compares received attribute sets. Returns:
|
|
- 0 if sets have no attributes for comparison
|
|
- 1 if sets are different
|
|
- 2 if sets are exactly the same
|
|
- 3 if first set is strictly better
|
|
- 4 if second set is strictly better
|
|
"""
|
|
if len(attrs1) == 0 and len(attrs2) == 0:
|
|
return 0
|
|
if set(attrs1) != set(attrs2):
|
|
return 1
|
|
if all(attrs1[aid] == attrs2[aid] for aid in attrs1):
|
|
return 2
|
|
if all(
|
|
(attrs1[aid] <= attrs2[aid] and attrHig[aid] == 0) or
|
|
(attrs1[aid] >= attrs2[aid] and attrHig[aid] == 1) or
|
|
(attrs1[aid] == attrs2[aid] and attrHig[aid] == 2)
|
|
for aid in attrs1
|
|
):
|
|
return 3
|
|
if all(
|
|
(attrs2[aid] <= attrs1[aid] and attrHig[aid] == 0) or
|
|
(attrs2[aid] >= attrs1[aid] and attrHig[aid] == 1) or
|
|
(attrs1[aid] == attrs2[aid] and attrHig[aid] == 2)
|
|
for aid in attrs1
|
|
):
|
|
return 4
|
|
return 1
|
|
|
|
print('finding replacements')
|
|
skillReqAttribs = {
|
|
182: 277,
|
|
183: 278,
|
|
184: 279,
|
|
1285: 1286,
|
|
1289: 1287,
|
|
1290: 1288}
|
|
skillReqAttribsFlat = set(skillReqAttribs.keys()).union(skillReqAttribs.values())
|
|
# Get data on type groups
|
|
# Format: {type ID: group ID}
|
|
typesGroups = {}
|
|
for row in tables['evetypes']:
|
|
typesGroups[row['typeID']] = row['groupID']
|
|
# Get data on type attributes
|
|
# Format: {type ID: {attribute ID: attribute value}}
|
|
typesNormalAttribs = {}
|
|
typesSkillAttribs = {}
|
|
for row in tables['dgmtypeattribs']:
|
|
attributeID = row['attributeID']
|
|
if attributeID in skillReqAttribsFlat:
|
|
typeSkillAttribs = typesSkillAttribs.setdefault(row['typeID'], {})
|
|
typeSkillAttribs[row['attributeID']] = row['value']
|
|
# Ignore these attributes for comparison purposes
|
|
elif attributeID in (
|
|
124, # mainColor
|
|
422, # techLevel
|
|
633, # metaLevel
|
|
1692 # metaGroupID
|
|
):
|
|
continue
|
|
else:
|
|
typeNormalAttribs = typesNormalAttribs.setdefault(row['typeID'], {})
|
|
typeNormalAttribs[row['attributeID']] = row['value']
|
|
# Get data on skill requirements
|
|
# Format: {type ID: {skill type ID: skill level}}
|
|
typesSkillReqs = {}
|
|
for typeID, typeAttribs in typesSkillAttribs.items():
|
|
typeSkillAttribs = typesSkillAttribs.get(typeID, {})
|
|
if not typeSkillAttribs:
|
|
continue
|
|
typeSkillReqs = typesSkillReqs.setdefault(typeID, {})
|
|
for skillreqTypeAttr, skillreqLevelAttr in skillReqAttribs.items():
|
|
try:
|
|
skillType = int(typeSkillAttribs[skillreqTypeAttr])
|
|
skillLevel = int(typeSkillAttribs[skillreqLevelAttr])
|
|
except (KeyError, ValueError):
|
|
continue
|
|
typeSkillReqs[skillType] = skillLevel
|
|
# Get data on attribute highIsGood flag
|
|
# Format: {type ID: 0 if high is bad, 1 if high is good, 2 if neither}
|
|
attrHig = {}
|
|
for row in tables['dgmattribs']:
|
|
attrHig[row['attributeID']] = 1 if row['highIsGood'] else 0
|
|
# As CCP data is not really consistent, do some overrides
|
|
attrHig[4] = 0 # mass
|
|
attrHig[161] = 0 # volume
|
|
# Also do some customizations; some attributes are neither good nor bad when they change
|
|
attrHig[128] = 2 # chargeSize
|
|
attrHig[1547] = 2 # rigSize
|
|
# As EVE affects various types mostly depending on their group or skill requirements,
|
|
# we're going to group various types up this way
|
|
# Format: {(group ID, frozenset(skillreq, type, IDs)): [type ID, {attribute ID: attribute value}]}
|
|
groupedData = {}
|
|
for row in tables['evetypes']:
|
|
typeID = row['typeID']
|
|
typeAttribs = typesNormalAttribs.get(typeID, {})
|
|
# Ignore stuff w/o attributes
|
|
if not typeAttribs:
|
|
continue
|
|
# We need only skill types, not levels for keys
|
|
typeSkillreqs = frozenset(typesSkillReqs.get(typeID, {}))
|
|
typeGroup = typesGroups[typeID]
|
|
groupData = groupedData.setdefault((typeGroup, typeSkillreqs), [])
|
|
groupData.append((typeID, typeAttribs))
|
|
# Format: {type ID: set(type IDs)}
|
|
same = {}
|
|
better = {}
|
|
# Now, go through composed groups and for every item within it find items which are
|
|
# the same and which are better
|
|
for groupData in groupedData.values():
|
|
for type1, type2 in itertools.combinations(groupData, 2):
|
|
comparisonResult = compareAttrs(type1[1], type2[1], attrHig)
|
|
# Equal
|
|
if comparisonResult == 2:
|
|
same.setdefault(type1[0], set()).add(type2[0])
|
|
same.setdefault(type2[0], set()).add(type1[0])
|
|
# First is better
|
|
elif comparisonResult == 3:
|
|
better.setdefault(type2[0], set()).add(type1[0])
|
|
# Second is better
|
|
elif comparisonResult == 4:
|
|
better.setdefault(type1[0], set()).add(type2[0])
|
|
# Put this data into types table so that normal process hooks it up
|
|
for row in tables['evetypes']:
|
|
typeID = row['typeID']
|
|
row['replaceSame'] = ','.join('{}'.format(tid) for tid in sorted(same.get(typeID, ())))
|
|
row['replaceBetter'] = ','.join('{}'.format(tid) for tid in sorted(better.get(typeID, ())))
|
|
|
|
data = {}
|
|
|
|
# Dump all data to memory so we can easely cross check ignored rows
|
|
for jsonName, cls in tables.items():
|
|
with open(os.path.join(jsonPath, '{}.json'.format(jsonName)), encoding='utf-8') as f:
|
|
tableData = json.load(f)
|
|
if jsonName in rowsInValues:
|
|
tableData = list(tableData.values())
|
|
if jsonName == 'icons':
|
|
tableData = convertIcons(tableData)
|
|
if jsonName == 'phbtraits':
|
|
tableData = convertTraits(tableData)
|
|
if jsonName == 'clonegrades':
|
|
tableData = convertClones(tableData)
|
|
data[jsonName] = tableData
|
|
|
|
fillReplacements(data)
|
|
|
|
# Set with typeIDs which we will have in our database
|
|
# Sometimes CCP unpublishes some items we want to have published, we
|
|
# can do it here - just add them to initial set
|
|
eveTypes = set()
|
|
for row in data['evetypes']:
|
|
if (row['published']
|
|
or row['groupID'] == 1306 # group Ship Modifiers, for items like tactical t3 ship modes
|
|
or row['typeName'].startswith('Civilian') # Civilian weapons
|
|
or row['typeID'] in (41549, 41548, 41551, 41550) # Micro Bombs (Fighters)
|
|
or row['groupID'] in (
|
|
1882,
|
|
1975,
|
|
1971,
|
|
1983 # the "container" for the abyssal environments
|
|
) # Abyssal weather (environment)
|
|
):
|
|
eveTypes.add(row['typeID'])
|
|
|
|
# ignore checker
|
|
def isIgnored(file, row):
|
|
if file in ('evetypes', 'dgmtypeeffects', 'dgmtypeattribs', 'invmetatypes') and row['typeID'] not in eveTypes:
|
|
return True
|
|
return False
|
|
|
|
# Loop through each json file and write it away, checking ignored rows
|
|
for jsonName, table in data.items():
|
|
fieldMap = fieldMapping.get(jsonName, {})
|
|
tmp = []
|
|
|
|
print('processing {}'.format(jsonName))
|
|
|
|
for row in table:
|
|
# We don't care about some kind of rows, filter it out if so
|
|
if not isIgnored(jsonName, row):
|
|
if jsonName == 'evetypes' and row['typeName'].startswith('Civilian'): # Apparently people really want Civilian modules available
|
|
row['published'] = True
|
|
|
|
instance = tables[jsonName]()
|
|
# fix for issue 80
|
|
if jsonName is 'icons' and 'res:/ui/texture/icons/' in str(row['iconFile']).lower():
|
|
row['iconFile'] = row['iconFile'].lower().replace('res:/ui/texture/icons/', '').replace('.png', '')
|
|
# with res:/ui... references, it points to the actual icon file (including it's size variation of #_size_#)
|
|
# strip this info out and get the identifying info
|
|
split = row['iconFile'].split('_')
|
|
if len(split) == 3:
|
|
row['iconFile'] = '{}_{}'.format(split[0], split[2])
|
|
if jsonName is 'icons' and 'modules/' in str(row['iconFile']).lower():
|
|
row['iconFile'] = row['iconFile'].lower().replace('modules/', '').replace('.png', '')
|
|
|
|
if jsonName is 'clonegrades':
|
|
if row['alphaCloneID'] not in tmp:
|
|
cloneParent = eos.gamedata.AlphaClone()
|
|
setattr(cloneParent, 'alphaCloneID', row['alphaCloneID'])
|
|
setattr(cloneParent, 'alphaCloneName', row['alphaCloneName'])
|
|
eos.db.gamedata_session.add(cloneParent)
|
|
tmp.append(row['alphaCloneID'])
|
|
|
|
for k, v in row.items():
|
|
if isinstance(v, str):
|
|
v = v.strip()
|
|
setattr(instance, fieldMap.get(k, k), v)
|
|
|
|
eos.db.gamedata_session.add(instance)
|
|
|
|
# quick and dirty hack to get this data in
|
|
with open(os.path.join(jsonPath, 'dynamicattributes.json'), encoding='utf-8') as f:
|
|
bulkdata = json.load(f)
|
|
for mutaID, data in bulkdata.items():
|
|
muta = eos.gamedata.DynamicItem()
|
|
muta.typeID = mutaID
|
|
muta.resultingTypeID = data['inputOutputMapping'][0]['resultingType']
|
|
eos.db.gamedata_session.add(muta)
|
|
|
|
for x in data['inputOutputMapping'][0]['applicableTypes']:
|
|
item = eos.gamedata.DynamicItemItem()
|
|
item.typeID = mutaID
|
|
item.applicableTypeID = x
|
|
eos.db.gamedata_session.add(item)
|
|
|
|
for attrID, attrData in data['attributeIDs'].items():
|
|
attr = eos.gamedata.DynamicItemAttribute()
|
|
attr.typeID = mutaID
|
|
attr.attributeID = attrID
|
|
attr.min = attrData['min']
|
|
attr.max = attrData['max']
|
|
eos.db.gamedata_session.add(attr)
|
|
|
|
eos.db.gamedata_session.commit()
|
|
|
|
# CCP still has 5 subsystems assigned to T3Cs, even though only 4 are available / usable. They probably have some
|
|
# old legacy requirement or assumption that makes it difficult for them to change this value in the data. But for
|
|
# pyfa, we can do it here as a post-processing step
|
|
eos.db.gamedata_engine.execute('UPDATE dgmtypeattribs SET value = 4.0 WHERE attributeID = ?', (1367,))
|
|
|
|
eos.db.gamedata_engine.execute('UPDATE invtypes SET published = 0 WHERE typeName LIKE \'%abyssal%\'')
|
|
|
|
# fix for #1722 until CCP gets their shit together
|
|
eos.db.gamedata_engine.execute('UPDATE invtypes SET typeName = \'Small Abyssal Energy Nosferatu\' WHERE typeID = ? AND typeName = ?', (48419, ''))
|
|
|
|
print()
|
|
for x in CATEGORIES_TO_REMOVE:
|
|
cat = eos.db.gamedata_session.query(eos.gamedata.Category).filter(eos.gamedata.Category.ID == x).first()
|
|
print ('Removing Category: {}'.format(cat.name))
|
|
eos.db.gamedata_session.delete(cat)
|
|
|
|
eos.db.gamedata_session.commit()
|
|
eos.db.gamedata_engine.execute('VACUUM')
|
|
|
|
print('done')
|
|
|
|
if __name__ == '__main__':
|
|
parser = argparse.ArgumentParser(description='This scripts dumps effects from an sqlite cache dump to mongo')
|
|
parser.add_argument('-d', '--db', required=True, type=str, help='The sqlalchemy connectionstring, example: sqlite:///c:/tq.db')
|
|
parser.add_argument('-j', '--json', required=True, type=str, help='The path to the json dump')
|
|
args = parser.parse_args()
|
|
|
|
main(args.db, args.json)
|
|
|