用Django Rest Framework和AngularJS开始你的项目
作者: seele52
发布时间:2015-07-29 11:30:55
用Django Rest Framework和AngularJS开始你的项目
作者:Kevin Stone
原帖:Getting Started with Django Rest Framework and AngularJS
原帖时间:2013-10-02
译序:虽然本文号称是"hello world式的教程"(这么长的hello world?!),内容上也确实是Django Rest Framework和AngularJS能做出来的相对最简单的东西了,但是鉴于Django, Django Rest Framework和AngularJS本身的复杂程度,本文还是比较适合对于这几个构架还是需要有一点基础性了解的小伙伴们来阅读。
正文开始:
ReSTful API已成为现代web应用的标配了,Django Rest Framework是一个基于django的强大ReST端开发框架。AngularJS是一个可用于构建复杂页面应用的现代javascript构架,专注于关注点分离(Separation of concerns)(MVC)及依赖注入(dependency injection),鼓励人们使用可维护(且可测试)的模组来组装富客户端。
在这篇博文中,我将一步步构建出一个AngularJS前端调用ReST API的示例项目,让我们来看看如何简单地结合前端和后端来构建一个复杂的应用。我将会大量的使用代码样例来展示整个过程和一些解决方案。这个Github里有用到的代码。
我们来构建一个Django示例项目
首先,我们来创建一个简单的照片共享应用(简陋版的Instagram)作为演示,让用户可以查看在站点上共享的所有照片。
所有这个项目需要的示例代码都在这里,设置项目环境看这里。在这个过程中,bower+grunt会帮你安装AngularJS和其他一些javascript组件。
搞定以后会有一些可供API演示用的fixture,包括几个用户([‘Bob‘, ‘Sally‘, ‘Joe‘, ‘Rachel‘]
),两条post([‘This is a great post‘, ‘Another thing I wanted to share‘]
)和一些示例照片。包含在示例代码中的Makefile会为你创建这些数据。
关于这个示例:
- 我会略过配置,构建和运行示例代码的细节部分,这个repository里的这个说明涵盖了这方面的细节,如果有啥问题请在Github上告诉我,我保证会搞定的。
- 自从我发现Coffee-Script又高效又可读(还有点Pythonic)以后,我就用它来写前端了。Grunt提供了一个把所有coffee script打包到一个
script.js
脚本中的task。
项目的数据层(Models)
我们的数据层相当简单,简单得跟Django的入门教程里那种差不多。我们总共有三个models,User
,Post
和Photo
。一个user可以有很多个post(和很多粉丝),一个post可以展示很多photo(像collection或者gallery那样)。每个post有一个标题和一个可选描述。
from django.db import models from django.contrib.auth.models import AbstractUser class User(AbstractUser): followers = models.ManyToManyField(‘self‘, related_name=‘followees‘, symmetrical=False) class Post(models.Model): author = models.ForeignKey(User, related_name=‘posts‘) title = models.CharField(max_length=255) body = models.TextField(blank=True, null=True) class Photo(models.Model): post = models.ForeignKey(Post, related_name=‘photos‘) image = models.ImageField(upload_to="%Y/%m/%d")
基于Django Rest Framework的API
Django Rest Framework (DRF)提供了一个清晰的构架,你既可以用它来构建简单的一站式API构架,也可以做出复杂的ReST结构。其Serializer
,可以把model映射到其对应的一般化表示(JSON, XML之类的),其基于类的视图的扩展集也正符合API接口(endpoint)的需要。Serializer
和基于类的视图扩展的清晰的分离是其精妙之处。同时,你也可以自定义URL而无需依赖自动生成,这也是DRF和其他构架(比如Tastypie或者是Piston)不同的地方。这些框架model到API接口的转换基本上都是自动的,这也就造成了灵活性的降低,限制了其在不同情况的使用(尤其是在认证和嵌套资源(nested resources)方面)。
Model Serializers
DRF中的Serializer
主要负责把Django中model的实例表示为他们的API,在这其中,我们可以转换任意的数据类型,或者是为model提供额外的信息。拿user来说,我们可以只使用用其中的部分字段,而不包括password
和email
之类包含用户隐私的属性;或者拿photo来说,我们可以返回一个ImageField
作为图片的URL地址,而不是其相对于目录的位置。
比如我们想要直接从Post中获取user的信息(而不是一般情况下提供一个超链接),我们就可以如下定义一个PostSerializer
。这使得user的信息在客户端很容易获取,而不需要在获取每个post时都调用额外的API再来获取user。使用超链接的方法用注释写在下面作为对比。序列化的好处在你可以通过超链接来扩展来衍生的不同序列,而无需使用嵌套(比如我们要呈现用户订阅的源里的post列表)。
PostSerializer
中的author信息,是由author的API视图提供的,因此我们要在serializer中将其设为可选(required=False
),并将其添加到验证排除(validation exclusion)里。
from rest_framework import serializers from .models import User, Post, Photo class UserSerializer(serializers.ModelSerializer): posts = serializers.HyperlinkedIdentityField(‘posts‘, view_name=‘userpost-list‘, lookup_field=‘username‘) class Meta: model = User fields = (‘id‘, ‘username‘, ‘first_name‘, ‘last_name‘, ‘posts‘, ) class PostSerializer(serializers.ModelSerializer): author = UserSerializer(required=False) photos = serializers.HyperlinkedIdentityField(‘photos‘, view_name=‘postphoto-list‘) # author = serializers.HyperlinkedRelatedField(view_name=‘user-detail‘, lookup_field=‘username‘) def get_validation_exclusions(self): # Need to exclude `author` since we‘ll add that later based off the request exclusions = super(PostSerializer, self).get_validation_exclusions() return exclusions + [‘author‘] class Meta: model = Post class PhotoSerializer(serializers.ModelSerializer): image = serializers.Field(‘image.url‘) class Meta: model = Photo
好嘞,现在我们的示例fixture已经载入了,我们来试试这些serializers吧。你可能会看到一个DeprecationWarning
,这是因为我们使用了HyperlinkedIdentityField
,但是却没有提供一个请求对象来构建URL地址。实际上我们已经提供了,所以我们可以无视之。
>>> user = User.objects.get(username=‘bob‘) >>> from example.api.serializers import * >>> serializer = UserSerializer(user) >>> serializer.data {‘id‘: 2, ‘username‘: u‘bob‘, ‘first_name‘: u‘Bob‘, ‘last_name‘: u‘‘, ‘posts‘: ‘/api/users/bob/posts‘} >>> post = user.posts.all()[0] >>> PostSerializer(post).data {‘author‘: {‘id‘: 2, ‘username‘: u‘bob‘, ‘first_name‘: u‘Bob‘, ‘last_name‘: u‘‘, ‘posts‘: ‘/api/users/bob/posts‘}, ‘photos‘: ‘/api/posts/2/photos‘, u‘id‘: 2, ‘title‘: u‘Title #2‘, ‘body‘: u‘Another thing I wanted to share‘} >>> serializer = PostSerializer(user.posts.all(), many=True) >>> serializer.data [{‘author‘: {‘id‘: 2, ‘username‘: u‘bob‘, ‘first_name‘: u‘Bob‘, ‘last_name‘: u‘‘, ‘posts‘: ‘/api/users/bob/posts‘}, ‘photos‘: ‘/api/posts/2/photos‘, u‘id‘: 2, ‘title‘: u‘Title #2‘, ‘body‘: u‘Another thing I wanted to share‘}]
API的URL结构
对于我们的API结构,我们希望能够维持一个相对扁平的结构来对给定的资源定义一个规范的接口(canonical endpoints),但同时也希望能够为一些常用的过滤(比如某个给定user的post或者某个给定post的photo)提供一些方便的嵌套列表。注意我们使用model的主键作为其标识符,但对于user来说,我们使用他们的username,因为这也是一个独特的标识符(我们会在后面的视图中看到这个)
from django.conf.urls import patterns, url, include from .api import UserList, UserDetail from .api import PostList, PostDetail, UserPostList from .api import PhotoList, PhotoDetail, PostPhotoList user_urls = patterns(‘‘, url(r‘^/(?P<username>[0-9a-zA-Z_-]+)/posts$‘, UserPostList.as_view(), name=‘userpost-list‘), url(r‘^/(?P<username>[0-9a-zA-Z_-]+)$‘, UserDetail.as_view(), name=‘user-detail‘), url(r‘^$‘, UserList.as_view(), name=‘user-list‘) ) post_urls = patterns(‘‘, url(r‘^/(?P<pk>\d+)/photos$‘, PostPhotoList.as_view(), name=‘postphoto-list‘), url(r‘^/(?P<pk>\d+)$‘, PostDetail.as_view(), name=‘post-detail‘), url(r‘^$‘, PostList.as_view(), name=‘post-list‘) ) photo_urls = patterns(‘‘, url(r‘^/(?P<pk>\d+)$‘, PhotoDetail.as_view(), name=‘photo-detail‘), url(r‘^$‘, PhotoList.as_view(), name=‘photo-list‘) ) urlpatterns = patterns(‘‘, url(r‘^users‘, include(user_urls)), url(r‘^posts‘, include(post_urls)), url(r‘^photos‘, include(photo_urls)), )
API视图
Django Rest Framework的强大之处很大程度上在于其通用视图,有了它,可以在完全不用(或一点点)修改的情况下完成常规的增查改删(CRUD)。在最简单视图中,你只需要提供一个model
和一个serializer_class
以及一个扩展过的内建通用视图(比如ListAPIView
或者是RetrieveAPIView
)。
在我们的例子中,我们有两个自定义的部分。首先,对于user来说,我们希望使用username
作为查找字段而不是pk
。所以我们在视图中设置了lookup_field
(一般来说,这是url_kwarg
,也是模型中的字段名)
我们也希望能为我们的视图创建一个嵌套版本,可用于查看某个给定user的post或者是某个给定post内的photo,我们只要简单重写视图中的get_queryset
,就可以自定义参数滤过来queryset的结果(相对username
和pk
)
from rest_framework import generics, permissions from .serializers import UserSerializer, PostSerializer, PhotoSerializer from .models import User, Post, Photo class UserList(generics.ListCreateAPIView): model = User serializer_class = UserSerializer permission_classes = [ permissions.AllowAny ] class UserDetail(generics.RetrieveAPIView): model = User serializer_class = UserSerializer lookup_field = ‘username‘ class PostList(generics.ListCreateAPIView): model = Post serializer_class = PostSerializer permission_classes = [ permissions.AllowAny ] class PostDetail(generics.RetrieveUpdateDestroyAPIView): model = Post serializer_class = PostSerializer permission_classes = [ permissions.AllowAny ] class UserPostList(generics.ListAPIView): model = Post serializer_class = PostSerializer def get_queryset(self): queryset = super(UserPostList, self).get_queryset() return queryset.filter(author__username=self.kwargs.get(‘username‘)) class PhotoList(generics.ListCreateAPIView): model = Photo serializer_class = PhotoSerializer permission_classes = [ permissions.AllowAny ] class PhotoDetail(generics.RetrieveUpdateDestroyAPIView): model = Photo serializer_class = PhotoSerializer permission_classes = [ permissions.AllowAny ] class PostPhotoList(generics.ListAPIView): model = Photo serializer_class = PhotoSerializer def get_queryset(self): queryset = super(PostPhotoList, self).get_queryset() return queryset.filter(post__pk=self.kwargs.get(‘pk‘))
附:内建API浏览器
Django Rest Freamework的其中一个好处是,它自带了内建的API浏览器来测试我们的API,这与Django的admin模块非常像,这在刚开始开发的时候很有帮助。
只需要在浏览器里加载你的API接口,通过内容协商(content-negotiation),DRF就会为你呈现一个好使的客户端界面用来和你的API交互。
为API添加许可和从属关系
之前写的API视图允许任何人在我们的站上添加任何东西。使用Django Rest Framework的好处之一,是我们可以在不影响相关model和serializer的情况下,很容易在视图里添加许可控制。要让得到许可的用户才能修改我们的API资源,我们可以添加一些许可类来提供授权控制。这些许可类应该对请求返回布尔值。这使得我们可访问完整的请求,包括cookies和已认证用户等。
from rest_framework import permissions class SafeMethodsOnlyPermission(permissions.BasePermission): """Only can access non-destructive methods (like GET and HEAD)""" def has_permission(self, request, view): return self.has_object_permission(request, view) def has_object_permission(self, request, view, obj=None): return request.method in permissions.SAFE_METHODS class PostAuthorCanEditPermission(SafeMethodsOnlyPermission): """Allow everyone to list or view, but only the other can modify existing instances""" def has_object_permission(self, request, view, obj=None): if obj is None: # Either a list or a create, so no author can_edit = True else: can_edit = request.user == obj.author return can_edit or super(PostAuthorCanEditPermission, self).has_object_permission(request, view, obj)
除了简单的授权判断,我们还想在储存前添加用户信息。当有人创建了一个新的Post
,我们希望把当前用户赋值为author。因为PostList
和PostDetail
都要赋值author字段,所以我们可以写一个mixin类来处理这些通用配置(你也可以用ViewSets
来搞定这个)
class PostMixin(object): model = Post serializer_class = PostSerializer permission_classes = [ PostAuthorCanEditPermission ] def pre_save(self, obj): """Force author to the current user on save""" obj.author = self.request.user return super(PostMixin, self).pre_save(obj) class PostList(PostMixin, generics.ListCreateAPIView): pass class PostDetail(PostMixin, generics.RetrieveUpdateDestroyAPIView): pass
用AngularJS与你的API对接
随着更具可交互性的页面应用的普及,ReSTful API可以被富客户端界面更好的用于呈现你的应用数据模型,并与之交互。AngularJS拥有清晰的控制分离(separation of controls),所以是一个很好的选择。AngularJS的模块化结构需要一点点配置。你的APP的功能部分由定义了service,directive和controller的模块组成,这样可以获得更清晰的分离。
AngularJS的好处之一,是其提供了类javascript表达的响应式编程。我们可以在模板中简单引用一个变量,我们的页面就会随着变量的变化自动刷新。
对于我们的hello world式的教程来说,我们就简单把我们的post在示例应用中列出来。这是一个AngularJS的基本模板。首先,最顶层的是body
标签,我们指定在这也运行的angular app(也就是example.app.basic
),我们会在根模块中定义它。其次,我们需要指定一个controller来控制我们代码段(AppController)。AngularJS中的controller更像是传统的MVC中的model+controller(注入的$scope
就包括了model层),controller定义的scope包含了可嵌套的model实例,嵌套的层次随着DOM结构逐级向下。最后,我们使用了一个Angular的directive(ng-repeat
),这是一个用来遍历我们的储存在$state
中的post
的控制结构。在这个遍历中,我们用Angular的表达语法(类似于django的模板标签)定义了一些标签来输出作者的username
以及post
的标题和内容。
提示:使用verbatim标签来包裹AngularJS表达式来避免django的渲染。
提示:我把一些代码段放在了Django的 {% block %}
标签中,这用我就能在不同页面中扩展这个模板
{% load staticfiles %} <html> <head> <link rel="stylesheet" type="text/css" href="{% static "bootstrap/dist/css/bootstrap.css" %}"> </head> <body ng-app="{% block ng_app %}example.app.static{% endblock %}"> <div class="content" ng-controller="{% block ng_controller %}AppController{% endblock %}">{% block content %} {% verbatim %} <div class="panel" ng-repeat="post in posts"> <div class="panel-heading clearfix"> <h3 class="panel-title">{{ post.title }}</h3> <author class="pull-right">{{ post.author.username }}</author> </div> <p class="well">{{ post.body }}</p> </div> {% endverbatim %} {% endblock %}</div> <script src="{% static "underscore/underscore.js" %}"></script> <script src="{% static "angular/angular.js" %}"></script> <script src="{% static "angular-resource/angular-resource.js" %}"></script> <script src="{% static "js/script.js" %}"></script> </body> </html>
现在,我们用一个简单的controller为这个模板提供post的列表。我们暂时用硬编码来写post,在下一步我们会用Ajax来获取这些post。
app = angular.module ‘example.app.static‘, [] app.controller ‘AppController‘, [‘$scope‘, ‘$http‘, ($scope, $http) -> $scope.posts = [ author: username: ‘Joe‘ title: ‘Sample Post #1‘ body: ‘This is the first sample post‘ , author: username: ‘Karen‘ title: ‘Sample Post #2‘ body: ‘This is another sample post‘ ] ]
用XHR从API获取Post
现在我们进一步更新我们的controller,让它从我们的API接口获取post的列表。AngularJS中提供的$http
服务和jQuery的$.ajax
或者其他的XHR部件很相似。注意,在AngularJS中,我们只需要用简单地用ajax请求得到的结果更新我们的model($scope.posts
),我们的视图也就会同步地被替换,不需要DOM操作。这种响应模式允许我们开发与数据模型分离的复杂UI,同时能使得UI组件能在无需介入连接细节的情况下能够正确地响应,使得我们能保持视图层和模型层的松耦合。
提示:注意$http
中承诺链式(promise chaining)调用(.then()
)的使用,这样可以很容易地传递回调。你也可以用链式调用的这个有点来写出更复杂的API工作流,比如创建一个新的post,同时保存3张照片
app = angular.module ‘example.app.basic‘, [] app.controller ‘AppController‘, [‘$scope‘, ‘$http‘, ($scope, $http) -> $scope.posts = [] $http.get(‘/api/posts‘).then (result) -> angular.forEach result.data, (item) -> $scope.posts.push item ]
$http
现在从我们的API获取post的列表,这样我们的例子现在就能展示服务器源的post。
利用Angular-Resource获取API
我们使用$http
来发起XHR调用来获取API的数据时,硬编码了许多API细节相关的代码,其中包括URL地址,http动作和其他一些本应包裹在更高层结构里的东西。通过Angular-Resource提供的机制,允许我们将API定义为Angular的service。通过Angular service来管理将更低层的http请求,可以使我们简单地使用ReSTful的动词来与API交互。
要使用Angular-Resource(也就是ngResource),我们只需要简单地映射把API接口映射到URL样式的参数上(和Django的urlpatterns
)很像。但不幸的是,要在Django和ngResource的定义之间转换并不是那么的容易,所以这里没有那么的DRY。
当你使用$resource
定义你的resource时,你可以简单提供一个url样式、一个默认参数的映射表,外加一些可选的http方法。在我们的例子中,我们希望用User
resource实例的username
字段获取URL地址中的:username
参数。Post
和Photo
实例则直接使用作为主键的id
字段。
app = angular.module ‘example.api‘, [‘ngResource‘] app.factory ‘User‘, [‘$resource‘, ($resource) -> $resource ‘/api/users/:username‘, username: ‘@username‘ ] app.factory ‘Post‘, [‘$resource‘, ($resource) -> $resource ‘/api/posts/:id‘, id: ‘@id‘ ] app.factory ‘Photo‘, [‘$resource‘, ($resource) -> $resource ‘/api/photos/:id‘, id: ‘@id‘ ]
现在我们定义的我们的API模块可以作为一个独立的部分在控制器模块中使用。我们可以将这些API resource做为service注入到我们的控制器中来访问API。我们将example.api
作为独立的模块添加进来,并且将所需的API resource作为控制器的依赖定义。resource默认提供了很多的基本的CRUD方法,包括query()
(用来获取集合),get()
(用来获取单独的元素),save()
和delete()
等。
app = angular.module ‘example.app.resource‘, [‘example.api‘] app.controller ‘AppController‘, [‘$scope‘, ‘Post‘, ($scope, Post) -> $scope.posts = Post.query() ]
结果见例子,和之前的Post列表相同。
为Post添加Photo
我们现在可以用ngResource
的API调用来显示我们的post了,但是现在只有一个调用来获取数据。在实际的应用里,你的数据几乎不会仅仅通过一个资源接口来调用,这就需要API调用的协同来构建你的应用的模型层。我们来改进我们的APP,这样我们就能在每个post里显示和浏览照片了。
首先,我们再添加了两个嵌套的API调用:
app.factory ‘UserPost‘, [‘$resource‘, ($resource) -> $resource ‘/api/users/:username/posts/:id‘ ] app.factory ‘PostPhoto‘, [‘$resource‘, ($resource) -> $resource ‘/api/posts/:post_id/photos/:id‘ ]
这会再为我们提供两个service(UserPost
和PostPhoto
),这样我们就可以相对于user和post来获取资源。我们希望能将嵌套的资源整合在AngluarJS中,使得基本资源加载后就加载他们(另一个方法是使用Angular的$watch
机制来响应变化,并触发额外的API调用)。在这儿,我们使用$q
服务提供的Promise/Deferred部件来进行链式调用。从ngResource-1.1+开始,任何提供了$promise
属性的资源都是可以被链的。我们用这种方式继续发起API调用来获取post的photo。
对于如何处理嵌套的资源,你有两个选择。在这里,我们就用post_id
作为标识符,为photo创建另一个$scope的容器。Anguilar的表达式和模板语言会忽略不存在的键,所以我们在模板里只要遍历photos[post.id]
来获取照片就可以了。注意,我们不需要显式地改变视图/模板,Angular的$digest
过程(也已经整合在ngResource
和$q
里了)可以检测更新。
app = angular.module ‘example.app.photos‘, [‘example.api‘] app.controller ‘AppController‘, [‘$scope‘, ‘Post‘, ‘PostPhoto‘, ($scope, Post, PostPhoto) -> $scope.photos = {} $scope.posts = Post.query() $scope.posts.$promise.then (results) -> # Load the photos angular.forEach results, (post) -> $scope.photos[post.id] = PostPhoto.query(post_id: post.id) ]
在模板中我们也做出修改来迭代这个model,从而为每个post显示phots。注意AngularJS是如何在API读取数据后再渲染的。在这里,我们对每个post中的每个photo对象通过id进行遍历。同时,我们使用了ng-src
而不是src
,这可以防止浏览器在计算angular表达式之前加载图片(不然你在日志里会看到404,无法找到”/media/{{ photo.image }}“)
<div class="panel" ng-repeat="post in posts"> <div class="panel-heading clearfix"> <h3 class="pull-left panel-title">{{ post.title }}</h3> <author class="pull-right">{{ post.author.username }}</author> </div> <p class="well">{{ post.body }}</p> <span class="photo" ng-repeat="photo in photos[post.id]"> <img class="img-thumbnail" ng-src="{{ photo.image }}"> </div> </div>
最后,我们就得到了带photo的示例页面。
附:AngularJS + CSRF保护
Django Rest Framework在使用SessionAuthentication
时(比如我们的例子中,在页面应用中使用同一个浏览器session),扩展了Django的跨站请求伪造(Cross Site Request Forgery)。这使得脚本在每次调用API时都要返回服务器提供的token。这有助于防止恶意脚本在用户不知情的情况下的访问。AngularJS的模块结构和依赖注入配置可以很容易地将CSRF Token包含在API请求的首部中(如果你愿意的话也可以放在cookie里)。
在我们的django模板中,只要简单地添加一个<script>
标签来配置$http
provider(这是$http
的依赖部件在AngularJS中的叫法),用Django的{{ csrf_token }}
模板变量为所有的API调用定义CSRF首部。
提示:确保这段脚本在你的模块定义之后加载
提示:你可以在Angular和Django间通过cookies或其他方式来传递CSRF token。在这里的显式调用只是为了确保csrf_token
在每个请求中都被生成了
<script> // Add the CSRF Token var app = angular.module(‘example.app‘); // Not including a list of dependent modules (2nd parameter to `module`) "re-opens" the module for additional configuration app.config([‘$httpProvider‘, function($httpProvider) { $httpProvider.defaults.headers.common[‘X-CSRFToken‘] = ‘{{ csrf_token|escapejs }}‘; }]); </script>
使用AngularJS创建和修改API资源
现在在我们的视图中创建一个编辑器来发布新的post(像Facebook里发布新的状态那样)。大多数的Angular教程都只是在添加页面上控制器的功能,我们想呈现的是如何保持控制器的简练和模块化,所以我们会为post的编辑器创建一个单独的控制器,并且展示控制器如何利用模型域的嵌套来扩展功能。这也给了我们一个机会来扩展现有的example.app.photos
模块,它提供了页面中基本的AppController
。
首先,我们要扩展我们的基本模板,为编辑器添加添加html模板。因为我们要使用非安全的方法来调用API保存新的post,所以我们还要添加CSRF token。
{% extends ‘base.html‘ %} {% block ng_app %}example.app.editor{% endblock %} {% block content %} {% verbatim %} <div ng-controller="EditController"> <h5>Create a New Post</h5> <form class="form-inline"> <div class="form-group block-level"> <input type="text" class="form-control" ng-model="newPost.title" placeholder="Title"> </div> <div class="form-group"> <input type="text" class="form-control" ng-model="newPost.body" placeholder="Body"> </div> <div class="form-group"> <button class="btn btn-default" ng-click="save()">Add Post</button> </div> </form> </div> {% endverbatim %} {{ block.super }} {% endblock %} {% block js %} {{ block.super }} <script> // Add the CSRF Token var app = angular.module(‘example.app.editor‘); // Not including a list of dependent modules (2nd parameter to `module`) "re-opens" the module for additional configuration app.config([‘$httpProvider‘, function($httpProvider) { $httpProvider.defaults.headers.common[‘X-CSRFToken‘] = ‘{{ csrf_token|escapejs }}‘; }]); </script> {% endblock %}
现在我们有了编辑器,还要写一个与之绑定的控制器。注意我们现在依赖两个模块,页面的基本模块和包含所有$resource
定义的API模块。
app = angular.module ‘example.app.editor‘, [‘example.api‘, ‘example.app.photos‘] app.controller ‘EditController‘, [‘$scope‘, ‘Post‘, ($scope, Post) -> $scope.newPost = new Post() $scope.save = -> $scope.newPost.$save().then (result) -> $scope.posts.push result .then -> # Reset our editor to a new blank post $scope.newPost = new Post() ]
之前,在API视图中,我们添加了一些许可限制以阻止其他用户修改某人的post。但之前,用户无论干什么其实都无所谓的。现在我们想发post(原文是create user,貌似说不通),就需要确保我们得作为一个有效的用户被授权,否则我们创建post的API请求就会被拒绝。在这个示例中,可以用Django的Authentication Backend自动作为管理员为你登陆。当然,不要在生产环境或者不受信任的环境下这么做。这只是用来帮助我们进行登陆和注册无关的沙箱测试的。
错误处理
跟着教程做的时候,你试过发一个没有title的post么?我们在Django的model中把这个作为必须字段了,所以Django Rest Framework在创建资源之前会验证这个字段。如果你试图创建一个没有title的Post
(无论是用API浏览器还是我们刚创建的表单),你都会得到一个400 Bad Request响应,并附上请求失败的原因。我们可以用这个来通知用户。
{
"title": [
"This field is required."
]
}
要通知用户,我们要修改我们的API调用。因为我们使用了Promise,我们可以简单地添加一个错误回调来捕捉响应,并且把它放在$scope
中,这样模板就会被更新,提示也就显示给了用户。
app = angular.module ‘example.app.editor‘, [‘example.api‘, ‘example.app.photos‘] app.controller ‘EditController‘, [‘$scope‘, ‘Post‘, ($scope, Post) -> $scope.newPost = new Post() $scope.save = -> $scope.newPost.$save().then (result) -> $scope.posts.push result .then -> # Reset our editor to a new blank post $scope.newPost = new Post() .then -> # Clear any errors $scope.errors = null , (rejection) -> $scope.errors = rejection.data ]
然后在模板上也加上一个便利的错误提示:
<p ng-repeat="(name, errs) in errors" class="alert alert-danger"><strong>{{ name }}</strong>: {{ errs.join(‘, ‘) }}</p>
只需要简单的把promise的API链在一起,就可以很直观地添加UI元素来提供过程的反馈(加载提示和进度条之类的)。AngularJS有相当完整的Promise规范来确保error(或者rejection)向下传递,使得处理error变得更简单。
就这个例子而言,我们只是将错误枚举出来放到bootstrap的alert box里。因为错误是以字段名作为标识符的,所以你可以很容易地在模板中将错误提示放在响应的表单字段附近。
删除Post
要完善我们的编辑器,我们还需要添加一个删除post的功能,使得当前用户可以删除自己post,我们已经构建的API可以防止用户来修改/删除不属于自己的资源。当你对AngularJS还不熟悉的时候,Angular的模块让人很难理解到底应该如何给controller提供初始数据。在我们的例子中,我们需要知道当前用户是谁,这样我们就能控制到底哪些post是可以被删除的。
这其中,要理解这一切的诀窍是尝试将你的controller分解成很多处理实际逻辑的service/factory(在angular里这俩差不多一个意思)。controller(很像Django里的view)只应该关注于整合不同的部件。在Django中,你应该在模型层中整合尽可能多的业务逻辑(胖模型),在Angular中,你也应该类似地将业务逻辑放到可组合的service中。
要添加删除post的功能,我们先要添加一个模块扩展我们的编辑器,并一个额外的控制器来处理删除事物。我们需要依赖一个叫AuthUser
的service,在这个service中,当前用户会由Django的模版中渲染。在本例中,我们就只简单把当前用户的username
属性渲染出来(如果没有登陆就留空)。我们在scope中添加两个函数,canDelete
来判断给定post是否能被user删除,delete
来删除post。这两个函数都以模板中提供的post
为参数。
我们在这里又一次用到了$resource
的Prominse接口,在成功的从服务器确认已删除后,我们才将post从站点视图的列表中删除。和之前一样,我们可以捕获错误结果并为用户提供反馈,在这里就略过了。
app = angular.module ‘example.app.manage‘, [‘example.api‘, ‘example.app.editor‘] app.controller ‘DeleteController‘, [‘$scope‘, ‘AuthUser‘, ($scope, AuthUser) -> $scope.canDelete = (post) -> return post.author.username == AuthUser.username $scope.delete = (post) -> post.$delete() .then -> # Remove it from the list on success idx = $scope.posts.indexOf(post) $scope.posts.splice(idx, 1) ]
在定义好控制器后,我们来更新我们的Post模板,使其在canDelete
为真时显示一个删除按钮。
{% extends ‘editor.html‘ %} {% block ng_app %}example.app.manage{% endblock %} {% block post_header %} <button type="button" class="close" ng-controller="DeleteController" ng-click="delete(post)" ng-show="canDelete(post)">×</button> {{ block.super }} {% endblock %} {% block js %} {{ block.super }} <script> // Configure the current user var app = angular.module(‘example.app.manage‘); // Not including a list of dependent modules (2nd parameter to `module`) "re-opens" the module for app.factory(‘AuthUser‘, function() { return { username: "{{ user.username|default:‘‘|escapejs }}" } }); </script> {% endblock %}
现在你加载示例页面,就能看到在root
的post上出现了一个`×`,但是bob
的帖子上没有。点×就会调用我们的API来删除post。
到此为止,我们就有了一个简单的分享源可以给用户分享消息了。
总结
好了,我们回过头来看看。用少量的代码(~100行的前端,~200行的后端),Django Rest Framework(当然还有Django本身)和AngularJS,我们得以很快速地创建一个简单的应用来发布一些简单的post,通过ReSTful的API层将Django的Django的数据模型基于用例导出,因为DRF而变得非常容易。AngularJS则使得我们非常容易地以更加结构化和模块化的方式与API交互,使得我们在添加新的功能的时候不会写出意大利面一样的代码。
本文所有的代码都可以在这个GitHub上找到。你可以自己checkout一个repository来继续你自己的尝试。如果发现任何错误,请留一个Issue(Pull-request就更好了),我保证我会更正的。如果你有任何问题,请给我留言(或者在Twitter上@kevinastone)。我会继续这一系列的文章来写更多的DRF+Angular的解决方案,包括:
- 分页
- 单列(singleton)接口(切换 关注/不关注)
- 更复杂的许可
- 丰富的验证
- 测试