-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgenerate_user_files.py
302 lines (266 loc) · 10.7 KB
/
generate_user_files.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
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
# -*- coding: utf-8 -*-
""" Script to create user files (user-config.py, user-fixes.py) """
#
# (C) Pywikipedia team, 2010-2012
#
__version__ = '$Id$'
#
# Distributed under the terms of the MIT license.
#
import codecs, os, re, sys
import platform
single_wiki_families = ['commons', 'wikidata']
def get_base_dir():
"""Return the directory in which user-specific information is stored.
This is determined in the following order -
1. If the script was called with a -dir: argument, use the directory
provided in this argument
2. If the user has a PYWIKIBOT2_DIR environment variable, use the value
of it
3. Use (and if necessary create) a 'pywikibot' folder (Windows) or
'.pywikibot' directory (Unix and similar) under the user's home
directory.
"""
# copied from config2.py, without the lines that check whether the
# directory already contains a user-config.py file
# this code duplication is nasty, should fix
NAME = "pywikibot"
for arg in sys.argv[1:]:
if arg.startswith("-dir:"):
base_dir = arg[5:]
sys.argv.remove(arg)
break
else:
if "PYWIKIBOT2_DIR" in os.environ:
base_dir = os.environ["PYWIKIBOT2_DIR"]
else:
is_windows = sys.platform == 'win32'
home = os.path.expanduser("~")
if is_windows:
_win_version = int(platform.version()[0])
if _win_version == 5:
base_dir = os.path.join(home, "Application Data", NAME)
elif _win_version == 6:
base_dir = os.path.join(home, "AppData\\Roaming", NAME)
else:
base_dir = os.path.join(home, "."+NAME)
if not os.path.isdir(base_dir):
os.makedirs(base_dir, mode=0700)
if not os.path.isabs(base_dir):
base_dir = os.path.normpath(os.path.join(os.getcwd(), base_dir))
return base_dir
base_dir = get_base_dir()
console_encoding = sys.stdout.encoding
# the directory in which generate_user_files.py is located
pywikibot_dir = sys.path[0]
if console_encoding is None or sys.platform == 'cygwin':
console_encoding = "iso-8859-1"
def listchoice(clist = [], message = None, default = None):
if not message:
message = "Select"
if default:
message += " (default: %s)" % default
message += ": "
for n, i in enumerate(clist):
print ("%d: %s" % (n + 1, i))
while True:
choice = raw_input(message)
if choice == '' and default:
return default
try:
return clist[int(choice) - 1]
except:
print("Invalid response")
return response
def change_base_dir():
"""Create a new user directory."""
global base_dir
while True:
new_base = raw_input("New user directory? ")
new_base = os.path.abspath(new_base)
if os.path.exists(new_base):
if os.path.isfile(new_base):
print("ERROR: there is an existing file with that name.")
continue
# make sure user can read and write this directory
if not os.access(new_base, os.R_OK|os.W_OK):
print("ERROR: directory access restricted")
continue
print("OK: using existing directory")
break
else:
try:
os.mkdir(new_base, 0700)
except Exception:
print("ERROR: directory creation failed")
continue
print("OK: Created new directory.")
break
from textwrap import wrap
msg = wrap("""WARNING: Your user files will be created in the directory
'%(new_base)s' you have chosen. To access these files, you will either have
to use the argument "-dir:%(new_base)s" every time you run the bot, or set
the environment variable "PYWIKIBOT2_DIR" equal to this directory name in
your operating system. See your operating system documentation for how to
set environment variables.""" % locals(), width=76)
for line in msg:
print line
ok = raw_input("Is this OK? ([yes], [N]o) ")
if ok in ["Y", "y"]:
base_dir = new_base
return True
print "Aborting changes."
return False
def file_exists(filename):
if os.path.exists(filename):
print("'%s' already exists." % filename)
return True
return False
def get_site_and_lang():
known_families = re.findall(r'(.+)_family.py\b',
'\n'.join(os.listdir(
os.path.join(pywikibot_dir,
"pywikibot",
"families"))))
known_families = sorted(known_families)
fam = listchoice(known_families,
"Select family of sites we are working on",
default='wikipedia')
if fam not in single_wiki_families:
mylang = raw_input(
"The language code of the site we're working on (default: 'en'): ") or 'en'
else:
mylang = fam
username = raw_input("Username (%s %s): "
% (mylang, fam))
username = unicode(username, console_encoding)
return fam, mylang, username
def create_user_config():
_fnc = os.path.join(base_dir, "user-config.py")
if not file_exists(_fnc):
fam, mylang, mainusername = get_site_and_lang()
mainusername = mainusername or "UnnamedBot"
while True:
choice = raw_input(
"Which variant of user_config.py:\n[S]mall or [E]xtended (with further information)? "
).upper()
if choice in "SE":
break
#
# I don't like this solution. Temporary for me.
#
# determine what directory this script (generate_user_files.py) lives in
install = os.path.dirname(os.path.abspath(sys.argv[0]))
# config2.py will be in the pywikibot/ directory
f = codecs.open(os.path.join(install, "pywikibot", "config2.py"),
"r", "utf-8")
cpy = f.read()
f.close()
res = re.findall("^(############## (?:LOGFILE|"
"INTERWIKI|"
"SOLVE_DISAMBIGUATION|"
"IMAGE RELATED|"
"TABLE CONVERSION BOT|"
"WEBLINK CHECKER|"
"DATABASE|"
"SEARCH ENGINE|"
"COPYRIGHT|"
"FURTHER) SETTINGS .*?)^(?=#####|# =====)",
cpy, re.MULTILINE | re.DOTALL)
config_text = '\n'.join(res)
f = codecs.open(_fnc, "w", "utf-8")
if choice == 'E':
f.write("""# -*- coding: utf-8 -*-
# This is an automatically generated file. You can find more configuration
# parameters in 'config.py' file.
# The family of sites we are working on. wikipedia.py will import
# families/xxx_family.py so if you want to change this variable,
# you need to write such a file.
family = '%s'
# The language code of the site we're working on.
mylang = '%s'
# The dictionary usernames should contain a username for each site where you
# have a bot account.
usernames['%s']['%s'] = u'%s'
%s""" % (fam, mylang, fam, mylang, mainusername, config_text))
else:
f.write("""# -*- coding: utf-8 -*-
family = '%s'
mylang = '%s'
usernames['%s']['%s'] = u'%s'
""" % (fam, mylang, fam, mylang, mainusername))
while(raw_input("Do you want to add any other projects? (y/N)").upper() == "Y"):
fam, mylang, username = get_site_and_lang()
username = username or mainusername
f.write("usernames['%(fam)s']['%(mylang)s'] = u'%(username)s'\n" % locals())
f.close()
print("'%s' written." % _fnc)
def create_user_fixes():
_fnf = os.path.join(base_dir, "user-fixes.py")
if not file_exists(_fnf):
f = codecs.open(_fnf, "w", "utf-8")
f.write(r"""# -*- coding: utf-8 -*-
#
# This is only an example. Don't use it.
#
fixes['example'] = {
'regex': True,
'msg': {
'_default':u'no summary specified',
},
'replacements': [
(ur'\bword\b', u'two words'),
]
}
""")
f.close()
print("'%s' written." % _fnf)
if __name__ == "__main__":
while True:
print('\nYour default user directory is "%s"' % base_dir)
ok = raw_input("How to proceed? ([K]eep [c]hange) ").upper().strip()
if (not ok) or "KEEP".startswith(ok):
break
if "CHANGE".startswith(ok):
if change_base_dir():
break
while True:
if os.path.exists(os.path.join(base_dir, "user-config.py")):
break
do_copy = raw_input(
"Do you want to copy user files from an existing pywikipedia installation? "
).upper().strip()
if do_copy and "YES".startswith(do_copy):
oldpath = raw_input("Path to existing wikipedia.py? ")
if not os.path.exists(oldpath):
print("ERROR: Not a valid path")
continue
if os.path.isfile(oldpath):
# User probably typed /wikipedia.py at the end, so strip it
oldpath = os.path.dirname(oldpath)
if not os.path.isfile(os.path.join(oldpath, "user-config.py")):
print("ERROR: no user_config.py found in that directory")
continue
newf = file(os.path.join(base_dir, "user-config.py"), "wb")
oldf = file(os.path.join(oldpath, "user-config.py"), "rb")
newf.write(oldf.read())
newf.close(); oldf.close()
if os.path.isfile(os.path.join(oldpath, "user-fixes.py")):
newfix = file(os.path.join(base_dir, "user-fixes.py"), "wb")
oldfix = file(os.path.join(oldpath, "user-fixes.py"), "rb")
newfix.write(oldfix.read())
newfix.close() ; oldfix.close()
elif do_copy and "NO".startswith(do_copy):
break
if not os.path.isfile(os.path.join(base_dir, "user-config.py")):
a = raw_input("Create user-config.py file? ([y]es, [N]o) ")
if a[:1] in ["Y", "y"]:
create_user_config()
else:
print("NOTE: user-config.py already exists in the directory")
if not os.path.isfile(os.path.join(base_dir, "user-fixes.py")):
a = raw_input("Create user-fixes.py file? ([y]es, [N]o) ")
if a[:1] in ["Y", "y"]:
create_user_fixes()
else:
print("NOTE: user-fixes.py already exists in the directory")