-
Notifications
You must be signed in to change notification settings - Fork 9
/
bottle_mongo.py
177 lines (135 loc) · 5.78 KB
/
bottle_mongo.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
# -*- coding: utf-8 -*-
"""Mongo support for Bottle.
This module provides a Bottle extension for supporting MongoDB for:
- injecting a MongoDB connection into handler functions
- converting PyMongo style returns from handlers into JSON
This module contains the following public classes:
- MongoPlugin -- The plugin for supporting handler functions.
"""
__all__ = ['MongoPlugin',]
try:
from json import dumps
except ImportError:
from simplejson import dumps
import inspect
from bottle import PluginError, response, JSONPlugin
import pymongo
if pymongo.version_tuple[0] >= 3:
from pymongo import MongoClient
MongoReplicaSetClient = MongoClient
else:
try:
# Backward compatibility with PyMongo 2.3 to 2.8
from pymongo import MongoClient, MongoReplicaSetClient
except ImportError:
# Backward compatibility with PyMongo 2.2
from pymongo import Connection as MongoClient
MongoReplicaSetClient = None
from pymongo.uri_parser import parse_uri
import bson.json_util
class MongoPlugin(object):
"""Mongo Plugin for Bottle.
Connect to a mongodb instance, and add a DB in a Bottle callback
Sample :
import bottle
from bottle.ext import mongo
app = bottle.Bottle()
plugin = mongo.MongoPlugin(uri="...", db="mydb", json_mongo=True)
app.install(plugin)
@app.route('/show/:item')
def show(item, mongodb):
doc = mongodb['items'].find({item:"item")})
return doc
uri : MongoDB hostname or uri
db : Database
json_mongo : Override Bottle serializer using Mongo one
keyword : Override parameter name in Bottle function.
post_create : Callback function to customize database obj after creation.
This constructor passes any optional parameter of the pymongo
Connection/MongoClient/MongoReplicaSetClient constructor.
If you are using PyMongo 2.3 or greater, connections to ReplicaSets are
available by passing in multiple nodes in the connection uri.
"""
api = 2
def get_mongo(self):
"""Return the mongo connection from the environment."""
if self.mongo_db:
return self.mongo_db
if len(self.uri_params['nodelist']) > 1 and MongoReplicaSetClient:
nodes = ','.join(['%s:%d' % n for n in self.uri_params['nodelist']])
client = MongoReplicaSetClient(nodes,
**self.uri_params['options'])
else:
client = MongoClient(self.uri_params['nodelist'][0][0],
self.uri_params['nodelist'][0][1],
**self.uri_params['options'])
db = client[self.uri_params['database']]
if self.uri_params['username']:
if not db.authenticate(self.uri_params['username'],
self.uri_params['password']):
raise PluginError('Cannot authenticate to MongoDB for '
'user: %s' % self.uri_params['username'])
if self.post_create:
db = self.post_create(db)
self.mongo_db = db
return self.mongo_db
def __init__(self, uri, db, keyword='mongodb', json_mongo=False,
post_create=None, **kwargs):
if not uri:
raise PluginError("MongoDB uri is required")
self.uri_params = parse_uri(uri)
if not len(self.uri_params['nodelist']):
raise PluginError("MongoDB hostname and port not configured")
self.uri_params['database'] = self.uri_params['database'] or db
if not self.uri_params['database']:
raise PluginError("MongoDB database name not configured")
self.uri_params['options'].update(kwargs)
self.keyword = keyword
self.json_mongo = json_mongo
self.mongo_db = None
self.name = "mongo:" + keyword
self.post_create = post_create
def __str__(self):
return "bottle_mongo.MongoPlugin(keyword=%r)" % (self.keyword)
def __repr__(self):
return "bottle_mongo.MongoPlugin(keyword=%r)" % (self.keyword)
def normalize_object(self, obj):
"""Normalize mongo object for json serialization."""
if isinstance(obj, dict):
if "_id" in obj:
obj["id"] = str(obj["_id"])
del obj["_id"]
if isinstance(obj, list):
for a in obj:
self.normalize_object(a)
def setup(self, app):
"""Called as soon as the plugin is installed to an application."""
for other in app.plugins:
if not isinstance(other, MongoPlugin):
continue
if other.keyword == self.keyword:
raise PluginError("Found another MongoDB plugin with "
"conflicting settings (non-unique keyword).")
# Remove builtin JSON Plugin
if self.json_mongo:
for other in app.plugins:
if isinstance(other, JSONPlugin):
app.uninstall(other)
return
def apply(self, callback, context):
"""Return a decorated route callback."""
args = inspect.getargspec(context.callback)[0]
# Skip this callback if we don't need to do anything
if self.keyword not in args:
return callback
def wrapper(*a, **ka):
ka[self.keyword] = self.get_mongo()
rv = callback(*a, **ka)
if self.json_mongo: # Override bottle JSON->String serializer
if isinstance(rv, dict):
self.normalize_object(rv)
json_response = dumps(rv, default=bson.json_util.default)
response.content_type = 'application/json'
return json_response
return rv
return wrapper