sql_todo.py 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. #!/bin/env python
  2. from flask import Flask
  3. from flask_restful import reqparse, abort, Api, Resource
  4. from flask_httpauth import HTTPBasicAuth
  5. from flask_sqlalchemy import SQLAlchemy
  6. import json
  7. # This "protects" paths with @auth.login_required
  8. auth = HTTPBasicAuth()
  9. # This is a very basic (and dumb) security model
  10. # We'll actually want the @auth.verify_password model,
  11. # which will let us use hashed passwords, etc.
  12. @auth.get_password
  13. def get_password(username):
  14. if username == 'admin':
  15. return '12345'
  16. return None
  17. @auth.error_handler
  18. def unauthorized():
  19. return abort(401, message='Unauthorized access')
  20. app = Flask(__name__)
  21. app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///todo.db'
  22. db = SQLAlchemy(app)
  23. api = Api(app)
  24. class Todo(db.Model):
  25. id = db.Column(db.Integer, primary_key=True)
  26. name = db.Column(db.String(20), unique=True, nullable=False)
  27. detail = db.Column(db.String(200), unique=False, nullable=False)
  28. def __repr__(self):
  29. return '<Todo {0}: {1}'.format(self.id, self.name)
  30. def serialize(self):
  31. return {'id': self.id, 'name': self.name, 'detail': self.detail}
  32. TODOS = {
  33. 'todo1': {'task': 'build an API'},
  34. 'todo2': {'task': '?????'},
  35. 'todo3': {'task': 'profit!'},
  36. }
  37. def abort_if_todo_doesnt_exist(todo_id):
  38. task = Todo.query.filter_by(id=todo_id).first()
  39. if task is None:
  40. abort(404, message="Todo {} doesn't exist".format(todo_id))
  41. return task
  42. # if todo_id not in TODOS:
  43. # abort(404, message="Todo {} doesn't exist".format(todo_id))
  44. parser = reqparse.RequestParser()
  45. parser.add_argument('task')
  46. # Todo
  47. # shows a single todo item and lets you delete a todo item
  48. # Yes! You can add the @auth.login_required to methods of the class. ! \o/ !
  49. class TodoPath(Resource):
  50. def get(self, todo_id):
  51. t = abort_if_todo_doesnt_exist(todo_id)
  52. return { 'id': t.id, 'name': t.name, 'detail': t.detail }
  53. def delete(self, todo_id):
  54. t = abort_if_todo_doesnt_exist(todo_id)
  55. db.session.delete(t)
  56. db.session.commit()
  57. return '', 204
  58. # @auth.login_required
  59. def put(self, todo_id):
  60. args = parser.parse_args()
  61. t = args['task']
  62. if t is None:
  63. abort(400, message="I need text for the task. task=...")
  64. task = Todo(name=todo_id, detail=t)
  65. db.session.add(task)
  66. db.session.commit()
  67. return task.id, 201
  68. # TodoList
  69. # shows a list of all todos, and lets you POST to add new tasks
  70. class TodoList(Resource):
  71. def get(self):
  72. tasks = Todo.query.all()
  73. return [t.serialize() for t in tasks]
  74. # return TODOS
  75. def post(self):
  76. args = parser.parse_args()
  77. todo_id = int(max(TODOS.keys()).lstrip('todo')) + 1
  78. todo_id = 'todo%i' % todo_id
  79. TODOS[todo_id] = {'task': args['task']}
  80. return TODOS[todo_id], 201
  81. ##
  82. ## Actually setup the Api resource routing here
  83. ##
  84. api.add_resource(TodoList, '/todos', '/todos/')
  85. # api.add_resource(TodoList, '/todos/')
  86. api.add_resource(TodoPath, '/todos/<todo_id>')
  87. if __name__ == '__main__':
  88. app.run(port=11022, debug=True)