2020-05-19 11:16:25 +02:00
|
|
|
import base64
|
|
|
|
import json
|
|
|
|
import re
|
2020-07-10 22:44:29 +02:00
|
|
|
import socket
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
import requests
|
|
|
|
|
|
|
|
from lib import Client
|
2020-08-15 19:11:06 +02:00
|
|
|
from patch_makers import DnsPatchMaker, WikimediaMessagesPatchMaker
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-05-30 16:51:55 +02:00
|
|
|
final_text = ''
|
|
|
|
gerrit_path = 'https://gerrit.wikimedia.org/g/'
|
|
|
|
client = Client.newFromCreds()
|
|
|
|
|
|
|
|
|
|
|
|
def add_text(a):
|
|
|
|
global final_text
|
|
|
|
final_text += a + '\n'
|
|
|
|
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-08-15 19:11:06 +02:00
|
|
|
def add_checklist(url, text, checked):
|
|
|
|
if checked:
|
|
|
|
add_text(' [x] [[{}|{}]]'.format(url, text))
|
|
|
|
else:
|
|
|
|
add_text(' [] [[{}|{}]]'.format(url, text))
|
|
|
|
|
|
|
|
|
2020-05-19 11:16:25 +02:00
|
|
|
def get_file_from_gerrit(path):
|
|
|
|
gerrit_url = 'https://gerrit.wikimedia.org/g/'
|
|
|
|
url = gerrit_url + '{0}?format=TEXT'.format(path)
|
2020-07-14 16:43:18 +02:00
|
|
|
r = requests.get(url)
|
|
|
|
if r.status_code == 200:
|
|
|
|
return base64.b64decode(r.text).decode('utf-8')
|
|
|
|
else:
|
|
|
|
return ''
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
|
2020-07-10 22:44:29 +02:00
|
|
|
def hostname_resolves(hostname):
|
|
|
|
try:
|
|
|
|
socket.gethostbyname(hostname)
|
|
|
|
except socket.error:
|
|
|
|
return False
|
|
|
|
return True
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-08-15 19:11:06 +02:00
|
|
|
|
2020-05-19 11:16:25 +02:00
|
|
|
def handle_special_wiki_apache(parts):
|
2020-08-15 19:11:06 +02:00
|
|
|
apache_file = get_file_from_gerrit(
|
|
|
|
'operations/puppet/+/production/modules/mediawiki/manifests/web/prod_sites.pp')
|
2020-05-19 11:16:25 +02:00
|
|
|
url = '.'.join(parts)
|
|
|
|
return url in apache_file
|
|
|
|
|
|
|
|
|
|
|
|
def post_a_comment(comment):
|
|
|
|
comment = 'Hello, I am helping on creating this wiki. ' + comment + \
|
|
|
|
' ^_^ Sincerely, your Fully Automated Resource Tackler'
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2020-08-15 19:11:06 +02:00
|
|
|
def handle_subticket_for_cloud(task_details, db_name):
|
|
|
|
hasSubtasks = client.getTaskSubtasks(task_details['phid'])
|
|
|
|
if hasSubtasks:
|
|
|
|
return
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-08-15 19:11:06 +02:00
|
|
|
client.createSubtask('The new wiki is going to be **public**.', [
|
|
|
|
'PHID-PROJ-hwibeuyzizzy4xzunfsk', # DBA
|
|
|
|
'PHID-PROJ-bj6y6ks7ampcwcignhce' # Data services
|
|
|
|
], task_details['phid'], 'Prepare and check storage layer for ' + db_name)
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
|
2020-08-02 18:45:41 +02:00
|
|
|
def get_dummy_wiki(shard, family):
|
|
|
|
if family == "wiktionary":
|
|
|
|
return {
|
2020-08-11 10:49:08 +02:00
|
|
|
"s3": "aawiki",
|
|
|
|
"s5": "mhwiktionary",
|
2020-08-02 18:45:41 +02:00
|
|
|
}.get(shard, "?????")
|
|
|
|
else:
|
|
|
|
return {
|
2020-08-02 18:48:16 +02:00
|
|
|
"s3": "aawiki",
|
2020-08-11 10:49:08 +02:00
|
|
|
"s5": "muswiki"
|
2020-08-02 18:45:41 +02:00
|
|
|
}.get(shard, "?????")
|
|
|
|
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-08-15 19:11:06 +02:00
|
|
|
def create_patch_for_wikimedia_messages(db_name, english_name, url, lang, bug_id):
|
|
|
|
if not english_name:
|
|
|
|
return
|
|
|
|
r = requests.get(
|
|
|
|
'https://gerrit.wikimedia.org/r/changes/?q=bug:{}+project:mediawiki/extensions/WikimediaMessages'.format(bug_id))
|
|
|
|
b = json.loads('\n'.join(r.text.split('\n')[1:]))
|
|
|
|
if b:
|
|
|
|
return
|
|
|
|
maker = WikimediaMessagesPatchMaker(
|
|
|
|
db_name, english_name, url, lang, bug_id)
|
|
|
|
maker.run()
|
|
|
|
|
|
|
|
|
|
|
|
def create_patch_for_dns(lang, bug_id):
|
|
|
|
r = requests.get(
|
|
|
|
'https://gerrit.wikimedia.org/r/changes/?q=bug:{}+project:operations/dns'.format(bug_id))
|
|
|
|
b = json.loads('\n'.join(r.text.split('\n')[1:]))
|
|
|
|
if b:
|
|
|
|
return
|
|
|
|
maker = DnsPatchMaker(lang, bug_id)
|
|
|
|
maker.run()
|
|
|
|
|
|
|
|
|
2020-05-30 16:51:55 +02:00
|
|
|
def hande_task(phid):
|
|
|
|
global final_text
|
|
|
|
final_text = ''
|
|
|
|
task_details = client.taskDetails(phid)
|
2020-07-14 16:43:51 +02:00
|
|
|
print('Checking T%s' % task_details['id'])
|
2020-08-15 19:11:06 +02:00
|
|
|
task_tid = 'T' + task_details['id']
|
2020-07-11 14:27:30 +02:00
|
|
|
add_text('\n\n------\n**Pre-install automatic checklist:**')
|
2020-08-15 19:11:06 +02:00
|
|
|
wiki_spec = {}
|
|
|
|
for case in re.findall(r'\n- *?\*\*(.+?):\*\* *?(.+)', task_details['description']):
|
|
|
|
wiki_spec[case[0].strip()] = case[1].strip()
|
|
|
|
language_code = wiki_spec.get('Language code')
|
2020-05-19 11:16:25 +02:00
|
|
|
if not language_code:
|
2020-07-11 14:27:30 +02:00
|
|
|
print('lang code not found, skipping')
|
2020-05-19 11:16:25 +02:00
|
|
|
return
|
2020-08-15 19:11:06 +02:00
|
|
|
url = wiki_spec.get('Site URL')
|
2020-05-19 11:16:25 +02:00
|
|
|
if not url:
|
2020-07-11 14:27:30 +02:00
|
|
|
print('url not found, skipping')
|
2020-05-19 11:16:25 +02:00
|
|
|
return
|
|
|
|
parts = url.split('.')
|
|
|
|
if len(parts) != 3 or parts[2] != 'org':
|
2020-07-11 14:27:30 +02:00
|
|
|
print('the url looks weird, skipping')
|
2020-05-19 11:16:25 +02:00
|
|
|
return
|
2020-08-15 19:11:06 +02:00
|
|
|
shard = wiki_spec.get('Shard', 'TBD')
|
2020-08-02 18:45:41 +02:00
|
|
|
|
|
|
|
shardDecided = shard != "TBD"
|
|
|
|
if shardDecided:
|
|
|
|
add_text(' [X] #DBA decided about the shard')
|
|
|
|
else:
|
|
|
|
add_text(' [] #DBA decided about the shard')
|
2020-07-10 22:24:21 +02:00
|
|
|
|
2020-07-10 22:44:29 +02:00
|
|
|
special = parts[1] == 'wikimedia'
|
2020-08-15 19:11:06 +02:00
|
|
|
dns_url = gerrit_path + 'operations/dns/+/master/templates/wikimedia.org' if special else gerrit_path + \
|
|
|
|
'operations/dns/+/master/templates/helpers/langlist.tmpl'
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-07-10 22:44:29 +02:00
|
|
|
dns = hostname_resolves(url)
|
2020-05-19 11:16:25 +02:00
|
|
|
if not dns:
|
2020-08-15 19:11:06 +02:00
|
|
|
if not special:
|
|
|
|
create_patch_for_dns(language_code, task_tid)
|
|
|
|
add_checklist(dns_url, 'DNS', dns)
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
if parts[1] == 'wikipedia':
|
2020-07-10 22:24:21 +02:00
|
|
|
db_name = parts[0].replace('-', '_') + 'wiki'
|
2020-05-19 11:16:25 +02:00
|
|
|
else:
|
2020-07-10 22:24:21 +02:00
|
|
|
db_name = parts[0].replace('-', '_') + parts[1]
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-08-15 19:11:06 +02:00
|
|
|
if not special and wiki_spec.get('Special', '').lower() != 'yes':
|
|
|
|
handle_subticket_for_cloud(task_details, db_name)
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
if special:
|
2020-08-15 19:11:06 +02:00
|
|
|
apache_url = gerrit_path + \
|
|
|
|
'operations/puppet/+/master/modules/mediawiki/manifests/web/prod_sites.pp'
|
2020-05-19 11:16:25 +02:00
|
|
|
if not handle_special_wiki_apache(parts):
|
|
|
|
apache = False
|
|
|
|
else:
|
|
|
|
apache = True
|
2020-08-15 19:11:06 +02:00
|
|
|
add_checklist(apache_url, 'Apache config', apache)
|
2020-05-19 11:16:25 +02:00
|
|
|
else:
|
|
|
|
apache = True
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text(' [x] Apache config (Not needed)')
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
langdb_url = 'https://raw.githubusercontent.com/wikimedia/language-data/master/data/langdb.yaml'
|
|
|
|
r = requests.get(langdb_url)
|
|
|
|
if re.search(r'\n *?' + language_code + ':', r.text):
|
|
|
|
langdb = True
|
2020-08-15 19:11:06 +02:00
|
|
|
add_text(
|
|
|
|
' [x] [[{}|Language configuration in language data repo]]'.format(langdb_url))
|
2020-05-19 11:16:25 +02:00
|
|
|
else:
|
|
|
|
langdb = False
|
2020-08-15 19:11:06 +02:00
|
|
|
add_text(
|
|
|
|
' [] [[{}|Language configuration in language data repo]]'.format(langdb_url))
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
core_messages_url = 'https://raw.githubusercontent.com/wikimedia/mediawiki/master/languages/messages/Messages{}.php'.format(
|
|
|
|
language_code[0].upper() + language_code[1:]
|
|
|
|
)
|
|
|
|
r = requests.get(core_messages_url)
|
|
|
|
if r.status_code == 200:
|
|
|
|
core_lang = True
|
|
|
|
else:
|
|
|
|
core_lang = False
|
2020-08-15 19:11:06 +02:00
|
|
|
add_checklist(core_messages_url,
|
|
|
|
'Language configuration in mediawiki core', core_lang)
|
|
|
|
|
2020-05-19 11:16:25 +02:00
|
|
|
path = 'mediawiki/extensions/WikimediaMessages/+/master/i18n/wikimediaprojectnames/en.json'
|
|
|
|
wikimedia_messages_data = get_file_from_gerrit(path)
|
|
|
|
wikimedia_messages_data = json.loads(wikimedia_messages_data)
|
|
|
|
if 'project-localized-name-' + db_name in wikimedia_messages_data:
|
|
|
|
wikimedia_messages_one = True
|
|
|
|
else:
|
|
|
|
wikimedia_messages_one = False
|
2020-08-15 19:11:06 +02:00
|
|
|
english_name = wiki_spec.get('Project name (English)')
|
|
|
|
create_patch_for_wikimedia_messages(
|
|
|
|
db_name, english_name, url, language_code, task_tid)
|
|
|
|
add_checklist(gerrit_path + path,
|
|
|
|
'Wikimedia messages configuration', wikimedia_messages_one)
|
2020-05-19 11:16:25 +02:00
|
|
|
url = 'https://en.wikipedia.org/wiki/MediaWiki:Project-localized-name-' + db_name
|
|
|
|
r = requests.get(url)
|
|
|
|
if 'Wikipedia does not have a' not in r.text:
|
|
|
|
wikimedia_messages_one_deployed = True
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text(' [x] [[{}|deployed]]'.format(url))
|
2020-05-19 11:16:25 +02:00
|
|
|
else:
|
|
|
|
wikimedia_messages_one_deployed = False
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text(' [] [[{}|deployed]]'.format(url))
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
if parts[1] == 'wikipedia':
|
|
|
|
path = 'mediawiki/extensions/WikimediaMessages/+/master/i18n/wikimediainterwikisearchresults/en.json'
|
|
|
|
search_messages_data = get_file_from_gerrit(path)
|
|
|
|
search_messages_data = json.loads(search_messages_data)
|
|
|
|
if 'search-interwiki-results-' + db_name in search_messages_data:
|
|
|
|
wikimedia_messages_two = True
|
|
|
|
else:
|
|
|
|
wikimedia_messages_two = False
|
2020-08-15 19:11:06 +02:00
|
|
|
add_checklist(gerrit_path + path,
|
|
|
|
'Wikimedia messages (interwiki search result) configuration', wikimedia_messages_two)
|
2020-05-19 11:16:25 +02:00
|
|
|
url = 'https://en.wikipedia.org/wiki/MediaWiki:Search-interwiki-results-' + db_name
|
|
|
|
r = requests.get(url)
|
|
|
|
if 'Wikipedia does not have a' not in r.text:
|
|
|
|
wikimedia_messages_two_deployed = True
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text(' [x] [[{}|deployed]]'.format(url))
|
2020-05-19 11:16:25 +02:00
|
|
|
else:
|
|
|
|
wikimedia_messages_two_deployed = False
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text(' [] [[{}|deployed]]'.format(url))
|
2020-05-19 11:16:25 +02:00
|
|
|
else:
|
|
|
|
wikimedia_messages_two = True
|
|
|
|
wikimedia_messages_two_deployed = True
|
2020-08-15 19:11:06 +02:00
|
|
|
add_text(
|
|
|
|
' [x] Wikimedia messages (interwiki search result) configuration (not needed)')
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-08-02 18:45:41 +02:00
|
|
|
if dns and apache and langdb and core_lang and wikimedia_messages_one and wikimedia_messages_one_deployed and wikimedia_messages_two and wikimedia_messages_two_deployed and shardDecided:
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text('**The Wiki is ready to be created.**')
|
2020-05-19 11:16:25 +02:00
|
|
|
else:
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text('**The creation is blocked until these part are all done.**')
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text('\n-------\n**Post install automatic checklist:**')
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
path = 'mediawiki/services/restbase/deploy/+/master/scap/vars.yaml'
|
|
|
|
restbase = get_file_from_gerrit(path)
|
2020-08-15 19:11:06 +02:00
|
|
|
add_checklist(gerrit_path + path, 'RESTbase', '.'.join(parts) in restbase)
|
|
|
|
|
2020-05-19 11:16:25 +02:00
|
|
|
path = 'mediawiki/services/cxserver/+/master/config/languages.yaml'
|
|
|
|
cxconfig = get_file_from_gerrit(path)
|
2020-08-15 19:11:06 +02:00
|
|
|
add_checklist(gerrit_path + path, 'CX Config',
|
|
|
|
'\n- ' + language_code in cxconfig)
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
path = 'analytics/refinery/+/master/static_data/pageview/whitelist/whitelist.tsv'
|
|
|
|
refinery_whitelist = get_file_from_gerrit(path)
|
2020-08-15 19:11:06 +02:00
|
|
|
add_checklist(gerrit_path + path, 'Analytics refinery',
|
|
|
|
'.'.join(parts[:2]) in refinery_whitelist)
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-08-15 19:11:06 +02:00
|
|
|
url = 'pywikibot/core/+/master/pywikibot/families/{}_family.py'.format(
|
|
|
|
parts[1])
|
2020-05-19 11:16:25 +02:00
|
|
|
pywikibot = get_file_from_gerrit(url)
|
2020-08-15 19:11:06 +02:00
|
|
|
add_checklist(gerrit_path + url, 'Pywikibot',
|
|
|
|
"'{}'".format(language_code) in pywikibot)
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
url = 'https://www.wikidata.org/w/api.php?action=help&modules=wbgetentities'
|
|
|
|
wikiata_help_page = requests.get(url).text
|
2020-08-15 19:11:06 +02:00
|
|
|
add_checklist(url, 'Wikidata', db_name in wikiata_help_page)
|
2020-05-19 11:16:25 +02:00
|
|
|
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text(' [] Import from Incubator')
|
|
|
|
add_text(' [] Clean up old interwiki links')
|
|
|
|
add_text('\n-------')
|
|
|
|
add_text('**Step by step commands**:')
|
2020-08-02 18:48:16 +02:00
|
|
|
dummy_wiki = get_dummy_wiki(shard, parts[1])
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text('On deploy1001:')
|
|
|
|
add_text('`cd /srv/mediawiki-staging/`')
|
|
|
|
add_text('`git fetch`')
|
|
|
|
add_text('`git log -p HEAD..@{u}`')
|
|
|
|
add_text('`git rebase`')
|
|
|
|
add_text('On mwmaint1002:')
|
|
|
|
add_text('`scap pull`')
|
|
|
|
add_text('`mwscript extensions/WikimediaMaintenance/addWiki.php --wiki={dummy} {lang} {family} {db} {url}`'.format(
|
2020-08-15 19:11:06 +02:00
|
|
|
dummy=dummy_wiki, lang=language_code, family=parts[1], db=db_name, url='.'.join(
|
|
|
|
parts)
|
2020-05-20 15:41:09 +02:00
|
|
|
))
|
2020-08-15 19:11:06 +02:00
|
|
|
summary = 'Creating {db_name} ({phab})'.format(
|
|
|
|
db_name=db_name, phab=task_tid)
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text('On deploy1001:')
|
2020-08-11 11:23:41 +02:00
|
|
|
if shard != "s3":
|
|
|
|
add_text('`scap sync-file wmf-config/db-eqiad.php "{}"`'.format(summary))
|
|
|
|
add_text('`scap sync-file wmf-config/db-codfw.php "{}"`'.format(summary))
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text('`scap sync-file dblists "{}"`'.format(summary))
|
|
|
|
add_text('`scap sync-wikiversions "{}"`'.format(summary))
|
2020-05-20 15:41:09 +02:00
|
|
|
if parts[1] == 'wikimedia':
|
2020-08-15 19:11:06 +02:00
|
|
|
add_text(
|
|
|
|
'`scap sync-file multiversion/MWMultiVersion.php "{}"`'.format(summary))
|
2020-05-30 16:51:55 +02:00
|
|
|
add_text('`scap sync-file static/images/project-logos/ "{}"`'.format(summary))
|
2020-07-29 19:26:55 +02:00
|
|
|
add_text('`scap sync-file wmf-config/InitialiseSettings.php "{}"`'.format(summary))
|
2020-05-30 16:51:55 +02:00
|
|
|
if parts[1] != 'wikimedia':
|
|
|
|
add_text('`scap sync-file langlist "{}"`'.format(summary))
|
|
|
|
add_text('`scap update-interwiki-cache`')
|
|
|
|
add_text('\n**End of automatic output**')
|
|
|
|
old_report = re.findall(
|
|
|
|
r'(\n\n------\n\*\*Pre-install automatic checklist:\*\*.+?\n\*\*End of automatic output\*\*\n)',
|
|
|
|
task_details['description'], re.DOTALL)
|
|
|
|
if not old_report:
|
|
|
|
print('old report not found, appending')
|
2020-08-15 19:11:06 +02:00
|
|
|
client.setTaskDescription(
|
|
|
|
task_details['phid'], task_details['description'] + final_text)
|
2020-05-30 16:51:55 +02:00
|
|
|
else:
|
|
|
|
if old_report[0] != final_text:
|
|
|
|
print('Updating old report')
|
|
|
|
client.setTaskDescription(task_details['phid'],
|
|
|
|
task_details['description'].replace(old_report[0], final_text))
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
open_create_wikis_phid = 'PHID-PROJ-kmpu7gznmc2edea3qn2x'
|
|
|
|
for phid in client.getTasksWithProject(open_create_wikis_phid, statuses=['open']):
|
|
|
|
hande_task(phid)
|
2020-05-19 11:16:25 +02:00
|
|
|
|
|
|
|
|
|
|
|
main()
|