2019-06-26 12:22:01 -05:00
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
|
|
# not use this file except in compliance with the License. You may obtain
|
|
|
|
# a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
# License for the specific language governing permissions and limitations
|
|
|
|
# under the License.
|
|
|
|
|
2017-01-04 10:40:09 -05:00
|
|
|
import datetime
|
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import os.path
|
2019-06-26 12:22:01 -05:00
|
|
|
import subprocess
|
2017-01-04 10:40:09 -05:00
|
|
|
|
|
|
|
from docutils.io import FileOutput
|
|
|
|
from docutils import nodes
|
|
|
|
from docutils.parsers import rst
|
|
|
|
import icalendar
|
2018-10-09 10:46:37 -04:00
|
|
|
from sphinx.util import logging
|
2017-01-04 10:40:09 -05:00
|
|
|
import yaml
|
|
|
|
|
2018-10-09 10:46:37 -04:00
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
|
2017-01-04 10:40:09 -05:00
|
|
|
|
|
|
|
class PendingICS(nodes.Element):
|
|
|
|
|
2019-06-26 12:22:01 -05:00
|
|
|
def __init__(self, data_source, series_name, data, last_update):
|
2017-01-04 10:40:09 -05:00
|
|
|
super(PendingICS, self).__init__()
|
|
|
|
self._data_source = data_source
|
|
|
|
self._series_name = series_name
|
|
|
|
self._data = data
|
2019-06-26 12:22:01 -05:00
|
|
|
self.last_update = last_update
|
2017-01-04 10:40:09 -05:00
|
|
|
|
|
|
|
|
|
|
|
class ICS(rst.Directive):
|
|
|
|
|
|
|
|
option_spec = {
|
|
|
|
'source': rst.directives.unchanged,
|
|
|
|
'name': rst.directives.unchanged,
|
|
|
|
}
|
|
|
|
has_content = False
|
2019-06-26 12:22:01 -05:00
|
|
|
last_update = ''
|
2017-01-04 10:40:09 -05:00
|
|
|
|
|
|
|
def _load_data(self, env, data_source):
|
|
|
|
rel_filename, filename = env.relfn2path(data_source)
|
2019-06-26 12:22:01 -05:00
|
|
|
|
|
|
|
# Attempt to get the last update time for our timestamp
|
|
|
|
try:
|
|
|
|
last_update = subprocess.check_output(
|
|
|
|
[
|
|
|
|
'git', 'log', '-n1', '--format=%ad',
|
|
|
|
'--date=format:%Y-%m-%d %H:%M:%S',
|
|
|
|
'--', filename,
|
|
|
|
]
|
|
|
|
).decode('utf-8').strip()
|
|
|
|
self.last_update = datetime.datetime.strptime(
|
|
|
|
last_update, '%Y-%m-%d %H:%M:%S')
|
|
|
|
LOG.info('[ics] Last update of %s was %s',
|
|
|
|
rel_filename, last_update)
|
2019-09-03 11:45:26 -05:00
|
|
|
except (subprocess.CalledProcessError, ValueError):
|
2019-06-26 12:22:01 -05:00
|
|
|
self.last_update = datetime.datetime.utcnow()
|
|
|
|
|
2017-01-04 10:40:09 -05:00
|
|
|
if data_source.endswith('.yaml'):
|
|
|
|
with open(filename, 'r') as f:
|
2019-06-26 12:22:01 -05:00
|
|
|
return yaml.safe_load(f)
|
2017-01-04 10:40:09 -05:00
|
|
|
elif data_source.endswith('.json'):
|
|
|
|
with open(filename, 'r') as f:
|
|
|
|
return json.load(f)
|
|
|
|
else:
|
|
|
|
raise NotImplementedError('cannot load file type of %s' %
|
|
|
|
data_source)
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
env = self.state.document.settings.env
|
|
|
|
|
|
|
|
try:
|
|
|
|
data_source = self.options['source']
|
|
|
|
except KeyError:
|
|
|
|
error = self.state_machine.reporter.error(
|
|
|
|
'No source set for ics directive',
|
|
|
|
nodes.literal_block(self.block_text, self.block_text),
|
|
|
|
line=self.lineno)
|
|
|
|
return [error]
|
|
|
|
|
|
|
|
try:
|
|
|
|
series_name = self.options['name']
|
|
|
|
except KeyError:
|
|
|
|
error = self.state_machine.reporter.error(
|
|
|
|
'No name set for ics directive',
|
|
|
|
nodes.literal_block(self.block_text, self.block_text),
|
|
|
|
line=self.lineno)
|
|
|
|
return [error]
|
|
|
|
|
|
|
|
data = self._load_data(env, data_source)
|
|
|
|
|
2019-06-26 12:22:01 -05:00
|
|
|
node = PendingICS(data_source, series_name, data, self.last_update)
|
2017-01-04 10:40:09 -05:00
|
|
|
node.document = self.state.document
|
|
|
|
return [node]
|
|
|
|
|
|
|
|
|
2017-01-06 15:17:44 -05:00
|
|
|
_global_calendar = icalendar.Calendar()
|
|
|
|
_global_calendar.add('prodid', '-//releases.openstack.org//EN')
|
2019-06-26 12:22:01 -05:00
|
|
|
_global_calendar.add('version', '2.0')
|
2017-01-06 15:17:44 -05:00
|
|
|
_global_calendar.add('X-WR-CALNAME', 'OpenStack Release Schedule')
|
|
|
|
|
|
|
|
|
2017-01-11 13:53:01 -05:00
|
|
|
def _format_description(node):
|
|
|
|
"Given a node, get its text and remove line breaks in paragraphs."
|
|
|
|
text = node.astext()
|
|
|
|
parts = text.split('\n\n')
|
|
|
|
return '\n\n'.join(p.replace('\n', ' ') for p in parts)
|
|
|
|
|
|
|
|
|
2017-01-04 10:40:09 -05:00
|
|
|
def doctree_resolved(app, doctree, docname):
|
|
|
|
builder = app.builder
|
|
|
|
|
|
|
|
for node in doctree.traverse(PendingICS):
|
|
|
|
|
|
|
|
series_name = node._series_name
|
|
|
|
data = node._data
|
|
|
|
|
2018-10-09 10:46:37 -04:00
|
|
|
LOG.info('building {} calendar'.format(series_name))
|
2017-01-04 10:40:09 -05:00
|
|
|
|
|
|
|
cal = icalendar.Calendar()
|
|
|
|
cal.add('prodid', '-//releases.openstack.org//EN')
|
|
|
|
cal.add('X-WR-CALNAME', '{} schedule'.format(series_name))
|
|
|
|
|
|
|
|
for week in data['cycle']:
|
|
|
|
if not week.get('name'):
|
|
|
|
continue
|
|
|
|
|
|
|
|
event = icalendar.Event()
|
|
|
|
|
2017-01-04 17:22:01 -05:00
|
|
|
summary = []
|
2017-01-04 10:40:09 -05:00
|
|
|
for item in week.get('x-project', []):
|
|
|
|
try:
|
|
|
|
# Look up the cross-reference name to get the
|
|
|
|
# section, then get the title from the first child
|
|
|
|
# node.
|
|
|
|
title = doctree.ids[item].children[0].astext()
|
|
|
|
except Exception as e:
|
|
|
|
# NOTE(dhellmann): Catching "Exception" is a bit
|
|
|
|
# ugly, but given the complexity of the expression
|
|
|
|
# above there are a bunch of ways things might
|
|
|
|
# fail.
|
2018-10-09 10:46:37 -04:00
|
|
|
LOG.info('could not get title for {}: {}'.format(item, e))
|
2017-01-04 10:40:09 -05:00
|
|
|
title = item
|
2017-01-04 17:22:01 -05:00
|
|
|
summary.append(title)
|
|
|
|
if summary:
|
|
|
|
summary_text = ' (' + '; '.join(summary) + ')'
|
|
|
|
else:
|
|
|
|
summary_text = ''
|
2017-01-06 15:17:44 -05:00
|
|
|
event.add(
|
|
|
|
'summary',
|
|
|
|
'{} {}{}'.format(series_name.title(),
|
|
|
|
week['name'],
|
|
|
|
summary_text),
|
|
|
|
)
|
2017-01-04 17:22:01 -05:00
|
|
|
|
2019-06-26 12:22:01 -05:00
|
|
|
event.add('dtstamp', node.last_update)
|
|
|
|
event.add('uid', '%s-%s' % (series_name, week.get('name')))
|
|
|
|
|
2017-01-04 17:22:01 -05:00
|
|
|
start = datetime.datetime.strptime(week['start'], '%Y-%m-%d')
|
|
|
|
event.add('dtstart', icalendar.vDate(start.date()))
|
|
|
|
|
|
|
|
# NOTE(dhellmann): ical assumes a time of midnight, so in
|
|
|
|
# order to have the event span the final day of the week
|
|
|
|
# we have to add an extra day.
|
|
|
|
raw_end = datetime.datetime.strptime(week['end'], '%Y-%m-%d')
|
|
|
|
end = raw_end + datetime.timedelta(days=1)
|
|
|
|
event.add('dtend', icalendar.vDate(end.date()))
|
|
|
|
|
|
|
|
# Look up the cross-reference name to get the
|
|
|
|
# section, then add the full description to the
|
|
|
|
# text.
|
|
|
|
description = [
|
2017-01-11 13:53:01 -05:00
|
|
|
_format_description(doctree.ids[item])
|
2017-01-04 17:22:01 -05:00
|
|
|
for item in week.get('x-project', [])
|
2019-09-03 11:45:26 -05:00
|
|
|
if item in doctree.ids
|
2017-01-04 17:22:01 -05:00
|
|
|
]
|
2017-01-04 10:40:09 -05:00
|
|
|
if description:
|
2017-01-04 17:22:01 -05:00
|
|
|
event.add('description', '\n\n'.join(description))
|
2017-01-04 10:40:09 -05:00
|
|
|
|
|
|
|
cal.add_component(event)
|
2017-01-06 15:17:44 -05:00
|
|
|
_global_calendar.add_component(event)
|
2017-01-04 10:40:09 -05:00
|
|
|
|
|
|
|
output_full_name = os.path.join(
|
|
|
|
builder.outdir,
|
|
|
|
docname + '.ics',
|
|
|
|
)
|
|
|
|
output_dir_name = os.path.dirname(output_full_name)
|
|
|
|
if not os.path.exists(output_dir_name):
|
|
|
|
os.makedirs(output_dir_name)
|
|
|
|
destination = FileOutput(
|
|
|
|
destination_path=output_full_name,
|
|
|
|
encoding='utf-8',
|
|
|
|
)
|
2018-10-09 10:46:37 -04:00
|
|
|
LOG.info('generating {}'.format(output_full_name))
|
2017-01-04 10:40:09 -05:00
|
|
|
destination.write(cal.to_ical())
|
|
|
|
|
|
|
|
# Remove the node that the writer won't understand.
|
|
|
|
node.parent.replace(node, [])
|
|
|
|
|
|
|
|
|
2017-01-06 15:17:44 -05:00
|
|
|
def build_finished(app, exception):
|
|
|
|
if exception is not None:
|
|
|
|
return
|
|
|
|
builder = app.builder
|
|
|
|
output_full_name = os.path.join(
|
|
|
|
builder.outdir,
|
|
|
|
'schedule.ics',
|
|
|
|
)
|
|
|
|
output_dir_name = os.path.dirname(output_full_name)
|
|
|
|
if not os.path.exists(output_dir_name):
|
|
|
|
os.makedirs(output_dir_name)
|
|
|
|
destination = FileOutput(
|
|
|
|
destination_path=output_full_name,
|
|
|
|
encoding='utf-8',
|
|
|
|
)
|
2018-10-09 10:46:37 -04:00
|
|
|
LOG.info('generating {}'.format(output_full_name))
|
2017-01-06 15:17:44 -05:00
|
|
|
destination.write(_global_calendar.to_ical())
|
|
|
|
|
|
|
|
|
2017-01-04 10:40:09 -05:00
|
|
|
def setup(app):
|
2018-10-09 10:46:37 -04:00
|
|
|
LOG.info('initializing ICS extension')
|
2017-01-04 10:40:09 -05:00
|
|
|
app.add_directive('ics', ICS)
|
|
|
|
app.connect('doctree-resolved', doctree_resolved)
|
2017-01-06 15:17:44 -05:00
|
|
|
app.connect('build-finished', build_finished)
|
2020-04-11 15:47:13 -05:00
|
|
|
return {
|
|
|
|
'parallel_read_safe': True,
|
|
|
|
'parallel_write_safe': True,
|
|
|
|
}
|