forked from miguelgrinberg/sqlalchemy-soft-delete
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathapp.py
More file actions
127 lines (95 loc) · 3.77 KB
/
app.py
File metadata and controls
127 lines (95 loc) · 3.77 KB
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
import os
from flask import Flask, request, url_for, jsonify
from flask_sqlalchemy import SQLAlchemy, BaseQuery
from flask_migrate import Migrate
basedir = os.path.abspath(os.path.dirname(__file__))
app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///' + os.path.join(
basedir, 'app.db')
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
db = SQLAlchemy(app)
migrate = Migrate(app, db)
class QueryWithSoftDelete(BaseQuery):
_with_deleted = False
def __new__(cls, *args, **kwargs):
obj = super(QueryWithSoftDelete, cls).__new__(cls)
obj._with_deleted = kwargs.pop('_with_deleted', False)
if len(args) > 0:
super(QueryWithSoftDelete, obj).__init__(*args, **kwargs)
return obj.filter_by(deleted=False) if not obj._with_deleted else obj
return obj
def __init__(self, *args, **kwargs):
pass
def with_deleted(self):
return self.__class__(self._only_full_mapper_zero('get'),
session=db.session(), _with_deleted=True)
def _get(self, *args, **kwargs):
# this calls the original query.get function from the base class
return super(QueryWithSoftDelete, self).get(*args, **kwargs)
def get(self, *args, **kwargs):
# the query.get method does not like it if there is a filter clause
# pre-loaded, so we need to implement it using a workaround
obj = self.with_deleted()._get(*args, **kwargs)
return obj if obj is None or self._with_deleted or not obj.deleted else None
def soft_delete(self):
objs = self.all()
for obj in objs:
obj.deleted = True
return objs
class User(db.Model):
__tablename__ = 'users'
id = db.Column(db.Integer, primary_key=True)
name = db.Column(db.String(128))
deleted = db.Column(db.Boolean(), default=False)
query_class = QueryWithSoftDelete
def to_dict(self):
return {'id': self.id, 'name': self.name,
'url': url_for('get_user', id=self.id)
if not self.deleted else None}
class Message(db.Model):
__tablename__ = 'messages'
id = db.Column(db.Integer, primary_key=True)
message = db.Column(db.String(256))
user_id = db.Column(db.Integer, db.ForeignKey('users.id'))
user = db.relationship('User')
def to_dict(self):
return {'id': self.id, 'message': self.message,
'url': url_for('get_message', id=self.id),
'user_url': url_for('get_user', id=self.user_id)
if not self.user.deleted else None}
@app.route('/users', methods=['POST'])
def new_user():
user = User(**request.get_json())
db.session.add(user)
db.session.commit()
return '', 201, {'Location': url_for('get_user', id=user.id)}
@app.route('/users', methods=['GET'])
def get_users():
users = User.query
return jsonify({'users': [u.to_dict() for u in users]})
@app.route('/users/<id>', methods=['GET'])
def get_user(id):
user = User.query.get_or_404(id)
return jsonify(user.to_dict())
@app.route('/users/<id>', methods=['DELETE'])
def delete_user(id):
if User.query.filter(id=id).soft_delete():
db.session.commit()
return '', 204
else:
return '', 404
@app.route('/users/<id>/messages', methods=['POST'])
def new_message(id):
user = User.query.get_or_404(id)
message = Message(user_id=user.id, **request.get_json())
db.session.add(message)
db.session.commit()
return '', 201, {'Location': url_for('get_message', id=message.id)}
@app.route('/messages')
def get_messages():
messages = Message.query
return jsonify({'messages': [m.to_dict() for m in messages]})
@app.route('/messages/<id>')
def get_message(id):
message = Message.query.get_or_404(id)
return jsonify(message.to_dict())