-
Notifications
You must be signed in to change notification settings - Fork 4
/
atlauncher_import.py
236 lines (194 loc) · 7.88 KB
/
atlauncher_import.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
# -*- coding: utf-8 -*-
"""
Li-aung "Lewis" Yip
Json port "leagris" Léa Gris
"""
import json
import os
import urllib
import unshortenit
from file_handling import fetch_url
from mod_file import Mod_File
from mod_pack import Mod_Pack
import re
URL_BASE = "http://download.nodecdn.net/containers/atl/"
def atlauncher_to_catherder(pack_name, pack_version, download_cache_folder,
install_folder):
# Returns a Mod_Pack object describing the given ATLauncher mod pack.
# This is exactly what is used in the ATlauncher code.
# Note \W is shorter, but also includes the underscore. We are trying
# to exactly match ATLauncher behaviour.
# Note 2: ATLauncher calls this "getSafeName()".
sn = re.sub("[^A-Za-z0-9]","",pack_name)
if pack_name != sn:
print "INFO - changing pack name {p1} to {p2}.".format(p1=pack_name,p2=sn)
pack_name = sn
os.chdir(download_cache_folder)
config_json_file_path = fetch_atlauncher_config_json(pack_name, pack_version)
# Process Configs.json and pull out list of files to download.
with open(config_json_file_path, 'r') as data_file:
jsondata = json.load(data_file)
mods = jsondata['mods']
libs = jsondata['libraries']
# From config file, pull out minecraft version number (need this for
# creating 'dependency' folder, i.e. ./mods/1.7.10/.
minecraft_version = jsondata['minecraft']
# Mod_Pack is a parent class which contains a list of Mod_Files.
mp = Mod_Pack(pack_name, pack_version, download_cache_folder,
install_folder, minecraft_version)
# Config zip is a special file, not included in the modpack's XML
# description.
if not ('noConfigs' in jsondata and jsondata['noConfigs'] == True):
mp.mod_files.append(atlauncher_config_zip(pack_name, pack_version))
# Build list of mod files that need to be downloaded.
for mod in mods:
f = mod_handler(mod, minecraft_version)
if f is not None:
mp.mod_files.append(f)
for lib in libs:
f = lib_handler(lib)
if f is not None:
mp.mod_files.append(f)
return mp
def mod_lib_handler(mol, mod_or_lib):
assert mod_or_lib in ("mod", "lib")
f = Mod_File()
f['download_url_primary'] = expand_atlauncher_url(mol['url'],
mol['download'])
# Some entries don't return an md5 hash...
if 'md5' in mol:
f['download_md5'] = mol['md5']
else:
f['download_md5'] = None
f['install_filename'] = mol['file']
if mod_or_lib == 'mod':
f['name'] = mol['name']
f['description'] = mol['description']
else:
f['name'] = "A library."
f['description'] = 'A library.'
return f
def mod_handler(mod, minecraft_version):
# See ATLauncher source - java\com\atlauncher\data\Mod.java
f = mod_lib_handler(mod, 'mod')
# 'server' attribute may be missing from XML. Default is 'yes'.
f['required_on_server'] = not ('server' in mod.keys()
and mod['server'] == False)
f['required_on_client'] = not ('client' in mod.keys()
and mod['client'] == False)
if 'optional' in mod.keys() and mod['optional'] == True:
f['optional?'] = True
f[
'install_optional?'] = True # TODO - replace with question prompt or share code support.
else:
f['optional?'] = False
f[
'install_optional?'] = True # No effect, apart from satisfying validate() assert
install_types_folders = {
'mods': 'mods/',
'forge': './',
'dependency': 'mods/' + minecraft_version + "/",
'ic2lib': 'mods/ic2/',
'flan': 'mods/Flan/', # not tested
'denlib': 'mods/denlib/', # not tested
'plugins': 'plugins/', # not tested
'coremods': 'coremods/', # not tested
'jarmod': 'jarmods/', # not tested
'disabled': 'disabledmods/', #not tested
'bin': 'bin/', #not tested
'natives': 'natives/' #not tested
}
t = mod['type']
fn = f['install_filename']
url = f['download_url_primary']
if t in install_types_folders.keys():
f['install_method'] = 'copy'
f['install_path'] = install_types_folders[t]
if t in ['forge', 'mcpc']:
f['special_actions'] = 'create_run_sh'
elif t == 'extract':
f['install_method'] = 'unzip'
e = mod['extractTo']
if e == 'root':
f['install_path'] = './'
elif e == 'mods':
f['install_path'] = 'mods/'
else:
print (
"WARNING - didn't know what to do with file {f} of type {t} extractto type {e}.".format(
f=fn, t=t,
e=e))
else:
print (
"WARNING - didn't know what to do with file {f} of type {t}.".format(
f=fn, t=t))
return None
print ('Adding mod {m} to download list - {u}'.format(m=fn, u=url))
return f
def lib_handler(lib):
f = mod_lib_handler(lib, 'lib')
f['optional?'] = False
f['install_optional?'] = True
f['install_method'] = 'copy'
if 'server' in lib.keys():
f['required_on_server'] = True
[dir_path, filename] = os.path.split(
'libraries/' + lib['server'])
f['install_path'] = dir_path
l = f['install_filename']
u = f['download_url_primary']
print (
'Adding library {l} to download list - {u}'.format(l=l, u=u))
else:
f['required_on_server'] = False
f['required_on_client'] = True
return f
def fetch_atlauncher_config_json(pack_name, pack_version):
config_json_url = "{u}packs/{pn}/versions/{pv}/Configs.json".format(
u=URL_BASE, pn=pack_name, pv=pack_version)
config_json_filename = "Configs_{pn}_{pv}.json".format(pn=pack_name,
pv=pack_version)
config_json_file_path = fetch_url(config_json_url, config_json_filename, None)
return config_json_file_path
def atlauncher_config_zip(pack_name, pack_version):
mf = Mod_File()
mf[
'download_url_primary'] = "{u}packs/{pn}/versions/{pv}/Configs.zip".format(
u=URL_BASE, pn=pack_name,
pv=pack_version)
mf['install_filename'] = "Configs_{pn}_{pv}.zip".format(pn=pack_name,
pv=pack_version)
mf['required_on_server'] = True
mf['required_on_client'] = True
mf['name'] = "Configs"
mf['install_method'] = 'unzip'
mf['install_path'] = './'
mf['optional?'] = False
mf['install_optional?'] = True
return mf
def expand_atlauncher_url(original_url, download_type):
if download_type == 'direct':
return original_url
elif download_type == 'server':
# Note, pathname2url for applying percent encoding -
# "Pams HarvestCraft 1.7.10c.jar" is an example of something that
# needs percent-encoding.
return URL_BASE + urllib.quote(original_url, safe='/')
elif download_type == 'browser':
if 'http://adf.ly' in original_url:
status = unshortenit.unshorten(original_url)
if status[1] == 200: # 200 = HTTP OK
print ('Unshortened {url1} to {url2}.'.format(url1=original_url,
url2=status[0]))
return status[0]
else:
return 'INVALID ADFLY LINK OR OTHER HTTP ERROR'
return 'Download file manually'
if __name__ == "__main__":
atl_pack = atlauncher_to_catherder(pack_name="Bevo's Tech Pack",
pack_version="BTP-11-Full",
download_cache_folder="D:/mc_test/cache-btp",
install_folder="D:/mc_test/install-btp")
atl_pack.install_server()