mirror of
https://github.com/Foundry376/Mailspring.git
synced 2025-01-23 16:37:49 +08:00
191 lines
5.5 KiB
Python
191 lines
5.5 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 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 'Hello World!'
|
||
|
|
||
|
|
||
|
@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:
|
||
|
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": time.mktime(t.start.timetuple()),
|
||
|
"end": time.mktime(t.end.timetuple()),
|
||
|
"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 'Success!'
|
||
|
|
||
|
|
||
|
def _create_email(event, etime):
|
||
|
sender = event.organizer.email #"send-availability@nylas.com"
|
||
|
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)
|
||
|
|
||
|
msg.headers['From'] = sender
|
||
|
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):
|
||
|
with open("mailgun-key.txt") as f:
|
||
|
print msg.to_string()
|
||
|
key = f.read()
|
||
|
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', '1.0')
|
||
|
|
||
|
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"])
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
app.run(port=8888, debug=True)
|