Mailspring/examples/N1-Send-Availability/backend/backend.py
Drew Regitsky 92820fc3d2 fix(examples): revamp/fix Send-Availability, change URL to live Heroku link
Summary:
Several fixes and updates the the Send Availability example package. Switches from
using `electron-safe-ipc` to using `protocol.RegisterStringProtocol` to communicate
with the child window. Changes the URLs in the package from localhost to our live
demo backend.

Test Plan: manual

Reviewers: bengotow

Reviewed By: bengotow

Differential Revision: https://phab.nylas.com/D2271
2015-11-19 18:16:38 -08:00

207 lines
6.1 KiB
Python

from flask import Flask, request, render_template
from flanker import mime
from icalendar import Calendar, Event
from icalendar.prop import vCalAddress, vText
import requests
from cStringIO import StringIO
from datetime import datetime
import pytz
import time
import os
import models as m
from session import session_scope
from sqlalchemy.orm.exc import NoResultFound
app = Flask(__name__)
m.init_db()
@app.route('/', methods=["GET", "POST"])
def hello_world():
return render_template("success.html")
@app.route('/register-events', methods=["POST"])
def register_events():
"""
Accepts new event data to be stored in the database, awaiting scheduling.
"""
data = request.get_json(force=True)
# Save new event details and times to database
with session_scope() as dbsession:
conflicts = [
dbsession.query(m.EventTime)
.filter(m.EventTime.key == tm['serverKey']).first()
for tm in data['times']
]
if any(n is not None for n in conflicts):
raise 'An event key conflicts with an existing key: %s' % conflicts
times = [_make_event_time(d) for d in data['times']]
attendees = [_make_attendee(d) for d in data['attendees']]
event = _make_event(data["event"])
event.times = times
event.attendees = attendees
dbsession.add(event)
return ''
@app.route('/event/<key>', methods=["GET"])
def load_event(key):
"""
Displays the scheduling page for this event, with time range for the
passed "key" param selected.
"""
with session_scope() as dbsession:
try:
etime = dbsession.query(m.EventTime).filter(m.EventTime.key == key).one()
except NoResultFound:
return render_template("bad_event_link.html")
event = etime.event
times = []
for t in event.times:
times.append({
"start": timestamp(t.start),
"end": timestamp(t.end),
"key": t.key
})
return render_template("show_event.html",
event=event,
selected_key=key,
times_json=times)
@app.route('/event/<key>', methods=["POST"])
def schedule(key):
"""
Schedules an event at the time range corresponding to the passed
"key" param, by sending an email to all participants containing an
attached ICS file.
"""
with session_scope() as dbsession:
try:
etime = dbsession.query(m.EventTime).filter(m.EventTime.key == key).one()
except NoResultFound:
return render_template("bad_event_link.html")
event = etime.event
msg = _create_email(event, etime)
_send_email(msg)
for t in event.times:
dbsession.delete(t)
for a in event.attendees:
dbsession.delete(a)
dbsession.commit()
dbsession.delete(event)
return render_template('success.html')
def _create_email(event, etime):
sender = event.organizer.email
html_body = render_template("event_email.html", event=event, time=etime)
text_body = render_template("event_email.txt", event=event, time=etime)
msg = mime.create.multipart('mixed')
ical_txt = _make_ics(event,etime)
body = mime.create.multipart('alternative')
body.append(
mime.create.text('plain', text_body),
mime.create.text('html', html_body),
mime.create.text('calendar; method=REQUEST', ical_txt, charset='utf8')
)
msg.append(
body,
mime.create.attachment('application/ics', ical_txt, filename='event.ics', disposition='attachment', charset='utf8')
)
msg.headers['From'] = "scheduler@nylas.com"
msg.headers['Reply-To'] = sender
msg.headers['Subject'] = "Invitation: {}".format(event.title)
msg.headers['To'] = ", ".join([a.email for a in event.attendees])
return msg
def _send_email(msg):
key = os.environ['MAILGUN_KEY']
requests.post(
"https://api.mailgun.net/v3/mg.nylas.com/messages.mime",
auth=("api", key),
data={"to": msg.headers["To"]},
files={"message": StringIO(msg.to_string())}
)
def _make_ics(event, etime):
cal = Calendar()
cal.add('prodid', 'N1-send-availability-package')
cal.add('version', '2.0')
cal.add('method', 'REQUEST') # also have PUBLISH or CANCEL
evt = Event()
evt.add('summary', event.title)
evt.add('location', event.location)
evt.add('description', event.description)
evt.add('dtstart', etime.start)
evt.add('dtend', etime.end)
evt.add('dtstamp', datetime.now())
evt['uid'] = '{timestamp}/{email}'.format(
timestamp=time.mktime(datetime.now().timetuple()),
email=event.organizer.email
)
evt.add('priority', 5)
organizer = vCalAddress('MAILTO:{}'.format(event.organizer.email))
organizer.params['cn'] = vText(event.organizer.name)
organizer.params['role'] = vText('CHAIR')
evt['organizer'] = organizer
for attendee in event.attendees:
atnd = vCalAddress('MAILTO:{}'.format(attendee.email))
atnd.params['cn'] = vText(attendee.name)
atnd.params['ROLE'] = vText('REQ-PARTICIPANT')
evt.add('attendee', atnd, encode=0)
cal.add_component(evt)
return cal.to_ical()
def _make_event_time(time_data):
return m.EventTime(start=datetime.utcfromtimestamp(int(time_data['start'])),
end=datetime.utcfromtimestamp(int(time_data['end'])),
key=time_data['serverKey'])
def _make_event(event_data):
return m.Event(title=event_data["title"],
description=event_data["description"],
location=event_data["location"])
def _make_attendee(attendee_data):
return m.Attendee(name=attendee_data["name"],
email=attendee_data["email"],
is_sender=attendee_data["isSender"])
def timestamp(dt):
'Return POSIX timestamp as float'
return (dt - datetime.utcfromtimestamp(0)).total_seconds()
if __name__ == '__main__':
app.run(port=8888, debug=True)