-
Notifications
You must be signed in to change notification settings - Fork 27
/
Copy pathmain.py
executable file
·234 lines (194 loc) · 8.65 KB
/
main.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
#!/usr/bin/env python
"""
:copyright: (c) 2011 Local Projects, all rights reserved
:license: Affero GNU GPL v3, see LICENSE for more details.
Main module for the Change by Us application. This is
where the magic begins. Routes are set here, database
connection initialized, web.py application started.
"""
import os
import sys
from os import environ
from framework.log import log
from framework.orm_holder import OrmHolder
from framework.session_holder import *
from framework.task_manager import *
from framework.image_server import *
#from giveaminute import models
# Due to the fact that we are utilizing some custom
# libraries, we add the lib path for import.
sys.path.append("lib/")
from lib import web
# Define all the routes for the applications
ROUTES = ( r'/admin/?([^/.]*)/?([^/.]*)/?([^/.]*)', 'controllers.admin.Admin',
r'/calendar/?([^/.]*)/?([^/.]*)/?([^/.]*)/?([^/.]*)', 'controllers.calendar.Calendar',
r'/cms/?([^/.]*)/?([^/.]*)', 'controllers.admin.Admin',
r'/create/?([^/.]*)', 'controllers.createProject.CreateProject',
r'/idea/?([^/.]*)', 'controllers.idea.Idea',
r'/join/?([^/.]*)/?([^/.]*)', 'controllers.join.Join',
r'/project/?([^/.]*)/?([^/.]*)/?([^/.]*)', 'controllers.project.Project',
r'/resource/?([^/.]*)/?([^/.]*)', 'controllers.resource.Resource',
r'/search/?([^/.]*)', 'controllers.search.Search',
r'/twilio/?([^/.]*)', 'controllers.sms.twilio.Twilio',
r'/useraccount/?([^/.]*)', 'controllers.user.UserAccount',
# RESTufl Resources
r'/rest/v1/needs/', 'controllers.rest.NeedsList',
r'/rest/v1/needs/(?P<id>\d+)/', 'controllers.rest.NeedInstance',
r'/rest/v1/needs/(?P<need_id>\d+)/volunteers/', 'controllers.rest.NeedVolunteerList',
r'/rest/v1/keywords/', 'controllers.rest.PopularKeywordList',
r'/rest/v1/events/', 'controllers.rest.EventList',
r'/rest/v1/events/(?P<id>\d+)/', 'controllers.rest.EventInstance',
r'/?([^/.]*)/?([^/.]*)', 'controllers.home.Home' )
def sessionDB():
"""
Gets the session database object. Database object is based from
web.py's database handling. This utilizes values found in the
config.yaml file.
"""
config = Config.get('database')
return web.database(dbn=config['dbn'], user=config['user'], pw=config['password'], db=config['db'], host=config['host'])
def enable_smtp():
"""
Enable SMTP support for the web.py email handling. This
uses config values found in config.yaml.
"""
try:
smtp_config = Config.get('email').get('smtp')
web.webapi.config.email_engine = 'smtp'
web.webapi.config.smtp_server = smtp_config.get('host')
web.webapi.config.smtp_port = smtp_config.get('port')
web.webapi.config.smtp_starttls = smtp_config.get('starttls')
web.webapi.config.smtp_username = smtp_config.get('username')
web.webapi.config.smtp_password = smtp_config.get('password')
except Exception, e:
log.info("ERROR: Exception when loading SMTP: %s" % e)
def enable_aws_ses():
"""
Enable AWS SES support for the web.py email handling. This
uses config values found in config.yaml.
"""
try:
ses_config = Config.get('email').get('aws_ses')
web.webapi.config.email_engine = 'aws'
web.webapi.config.aws_access_key_id = ses_config.get('access_key_id')
web.webapi.config.aws_secret_access_key = ses_config.get('secret_access_key')
except Exception, e:
log.info("ERROR: Exception when loading SES: %s" % e)
def basic_processor(handler):
from traceback import format_exception
try:
result = handler()
except Exception, e:
log.error("Unhandled exception: %s" % e)
# Do we want to continue even after exception?
raise
return handler()
def load_sqla(handler):
"""
Create a load hook and use sqlalchemy's ``scoped session``. This construct
places the ``sessionmaker()`` into a registry that maintains a single
``Session`` per application thread.
For more information see: http://webpy.org/cookbook/sqlalchemy
"""
##
# TODO: This should be `engine = models.get_db_engine()`. See the note in
# giveaminute.models for more information.
#
class OrmContextManager (object):
def __enter__(self):
log.debug("*** Loading the ORM")
self.orm = OrmHolder().orm
log.debug('***** loaded as %r' % self.orm)
return self.orm
def __exit__(self, e_type=None, e_val=None, e_tb=None):
# Since load_sqla is a processor, it catches bubbled-up exceptions
from traceback import format_exception
if e_type == web.HTTPError:
log.debug("*** web.HTTPError with the ORM")
log.warning(''.join(format_exception(e_type, e_val, e_tb)))
self.orm.commit()
elif e_type:
log.debug("*** Unhandled exception - check console logs for details")
log.error(''.join(format_exception(e_type, e_val, e_tb)))
OrmHolder.invalidate()
else:
log.debug("*** Finishing up with the ORM %r" % self.orm)
self.orm.commit()
with OrmContextManager() as orm:
result = handler()
return result
def main():
web.config.logfile = Config.get('logfile')
log.info("|||||||||||||||||||||||||||||||||||| SERVER START |||||||||||||||||||||||||||||||||||||||||||")
if Config.get('dev'):
web.config.debug = True
log.info("Debug: %s" % web.config.debug)
web.config.session_parameters['cookie_name'] = 'gam'
# TODO:
# Start with SES and fall-back to SMTP if both are enabled
if Config.get('email').get('smtp') and Config.get('email').get('aws_ses'):
import boto
try:
c = boto.connect_ses(
aws_access_key_id = Config.get('email').get('aws_ses').get('access_key_id'),
aws_secret_access_key = Config.get('email').get('aws_ses').get('secret_access_key'))
# TODO: Need to add proper exception handling or at least error reporting!
# Use raw_email since this allows for attachments
sendQuota = c.get_send_quota()["GetSendQuotaResponse"]["GetSendQuotaResult"]
# Check if we're close to the smtp quota. 10 seems like a good number
sentLast24Hours = sendQuota.get('SentLast24Hours')
if sentLast24Hours is None:
sentLast24Hours = 0
sentLast24Hours = int(float(sentLast24Hours))
max24HourSend = sendQuota.get('Max24HourSend')
if max24HourSend is None:
max24HourSend = 0
max24HourSend = int(float(max24HourSend))
if sentLast24Hours >= max24HourSend- 10:
enable_smtp()
else:
enable_aws_ses()
except Exception, e:
log.info(e)
log.info("ERROR: Email falling back to SMTP")
enable_smtp()
# Set the email configurations:
elif Config.get('email').get('smtp'):
enable_smtp()
elif Config.get('email').get('aws_ses'):
enable_aws_ses()
if web.webapi.config.email_engine not in ['aws', 'smtp']:
try:
raise Exception("ERROR: No valid email engine has been configured. Please check your configurations")
except Exception, e:
log.info(e)
# Add blitz.io route. We put into new var because of an odd behaviors
# where a changed ROUTES is not handled correctly.
try:
if Config.get('blitz_io').get('route') and Config.get('app_env') != 'live':
blitz_route = r'/%s/?([^/.]*)' % Config.get('blitz_io').get('route')
NEW_ROUTES = (blitz_route, 'controllers.blitz.Blitz') + ROUTES
else:
NEW_ROUTES = ROUTES
except KeyError:
NEW_ROUTES = ROUTES
except Exception, e:
log.error(e)
app = web.application(NEW_ROUTES, globals())
db = sessionDB()
SessionHolder.set(web.session.Session(app, web.session.DBStore(db, 'web_session')))
# WARNING:
# Adding new processors may cause duplicate insertions!
# The basic_processor has been disabled for this reason
# app.add_processor(basic_processor)
# Load SQLAlchemy
app.add_processor(load_sqla)
return app
# Main logic for the CBU application. Does some basic configuration,
# then starts the web.py application.
if __name__ == "__main__":
try:
app = main()
app.run()
except Exception, e:
log.info("ERROR: %s" % e)