views.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. import json
  2. from urllib.parse import urlencode
  3. from xml.dom.minidom import parseString
  4. from django.contrib.auth.decorators import login_required, permission_required
  5. from django.core import mail
  6. from django.core.exceptions import ValidationError
  7. from django.forms import fields
  8. from django.forms.forms import Form
  9. from django.http import (
  10. HttpResponse,
  11. HttpResponseBadRequest,
  12. HttpResponseNotAllowed,
  13. HttpResponseNotFound,
  14. HttpResponseRedirect,
  15. )
  16. from django.shortcuts import render
  17. from django.template import Context, Template
  18. from django.test import Client
  19. from django.utils.decorators import method_decorator
  20. from django.views.generic import TemplateView
  21. def get_view(request):
  22. "A simple view that expects a GET request, and returns a rendered template"
  23. t = Template("This is a test. {{ var }} is the value.", name="GET Template")
  24. c = Context({"var": request.GET.get("var", 42)})
  25. return HttpResponse(t.render(c))
  26. async def async_get_view(request):
  27. return HttpResponse(b"GET content.")
  28. def trace_view(request):
  29. """
  30. A simple view that expects a TRACE request and echoes its status line.
  31. TRACE requests should not have an entity; the view will return a 400 status
  32. response if it is present.
  33. """
  34. if request.method.upper() != "TRACE":
  35. return HttpResponseNotAllowed("TRACE")
  36. elif request.body:
  37. return HttpResponseBadRequest("TRACE requests MUST NOT include an entity")
  38. else:
  39. protocol = request.META["SERVER_PROTOCOL"]
  40. t = Template(
  41. "{{ method }} {{ uri }} {{ version }}",
  42. name="TRACE Template",
  43. )
  44. c = Context(
  45. {
  46. "method": request.method,
  47. "uri": request.path,
  48. "version": protocol,
  49. }
  50. )
  51. return HttpResponse(t.render(c))
  52. def put_view(request):
  53. if request.method == "PUT":
  54. t = Template("Data received: {{ data }} is the body.", name="PUT Template")
  55. c = Context(
  56. {
  57. "Content-Length": request.META["CONTENT_LENGTH"],
  58. "data": request.body.decode(),
  59. }
  60. )
  61. else:
  62. t = Template("Viewing GET page.", name="Empty GET Template")
  63. c = Context()
  64. return HttpResponse(t.render(c))
  65. def post_view(request):
  66. """A view that expects a POST, and returns a different template depending
  67. on whether any POST data is available
  68. """
  69. if request.method == "POST":
  70. if request.POST:
  71. t = Template(
  72. "Data received: {{ data }} is the value.", name="POST Template"
  73. )
  74. c = Context({"data": request.POST["value"]})
  75. else:
  76. t = Template("Viewing POST page.", name="Empty POST Template")
  77. c = Context()
  78. else:
  79. t = Template("Viewing GET page.", name="Empty GET Template")
  80. # Used by test_body_read_on_get_data.
  81. request.read(200)
  82. c = Context()
  83. return HttpResponse(t.render(c))
  84. def post_then_get_view(request):
  85. """
  86. A view that expects a POST request, returns a redirect response
  87. to itself providing only a ?success=true querystring,
  88. the value of this querystring is then rendered upon GET.
  89. """
  90. if request.method == "POST":
  91. return HttpResponseRedirect("?success=true")
  92. t = Template("The value of success is {{ value }}.", name="GET Template")
  93. c = Context({"value": request.GET.get("success", "false")})
  94. return HttpResponse(t.render(c))
  95. def json_view(request):
  96. """
  97. A view that expects a request with the header 'application/json' and JSON
  98. data, which is deserialized and included in the context.
  99. """
  100. if request.META.get("CONTENT_TYPE") != "application/json":
  101. return HttpResponse()
  102. t = Template("Viewing {} page. With data {{ data }}.".format(request.method))
  103. data = json.loads(request.body.decode("utf-8"))
  104. c = Context({"data": data})
  105. return HttpResponse(t.render(c))
  106. def view_with_header(request):
  107. "A view that has a custom header"
  108. response = HttpResponse()
  109. response.headers["X-DJANGO-TEST"] = "Slartibartfast"
  110. return response
  111. def raw_post_view(request):
  112. """A view which expects raw XML to be posted and returns content extracted
  113. from the XML"""
  114. if request.method == "POST":
  115. root = parseString(request.body)
  116. first_book = root.firstChild.firstChild
  117. title, author = [n.firstChild.nodeValue for n in first_book.childNodes]
  118. t = Template("{{ title }} - {{ author }}", name="Book template")
  119. c = Context({"title": title, "author": author})
  120. else:
  121. t = Template("GET request.", name="Book GET template")
  122. c = Context()
  123. return HttpResponse(t.render(c))
  124. def redirect_view(request):
  125. "A view that redirects all requests to the GET view"
  126. if request.GET:
  127. query = "?" + urlencode(request.GET, True)
  128. else:
  129. query = ""
  130. return HttpResponseRedirect("/get_view/" + query)
  131. def method_saving_307_redirect_query_string_view(request):
  132. return HttpResponseRedirect("/post_view/?hello=world", status=307)
  133. def method_saving_308_redirect_query_string_view(request):
  134. return HttpResponseRedirect("/post_view/?hello=world", status=308)
  135. def _post_view_redirect(request, status_code):
  136. """Redirect to /post_view/ using the status code."""
  137. redirect_to = request.GET.get("to", "/post_view/")
  138. return HttpResponseRedirect(redirect_to, status=status_code)
  139. def method_saving_307_redirect_view(request):
  140. return _post_view_redirect(request, 307)
  141. def method_saving_308_redirect_view(request):
  142. return _post_view_redirect(request, 308)
  143. def view_with_secure(request):
  144. "A view that indicates if the request was secure"
  145. response = HttpResponse()
  146. response.test_was_secure_request = request.is_secure()
  147. response.test_server_port = request.META.get("SERVER_PORT", 80)
  148. return response
  149. def double_redirect_view(request):
  150. "A view that redirects all requests to a redirection view"
  151. return HttpResponseRedirect("/permanent_redirect_view/")
  152. def bad_view(request):
  153. "A view that returns a 404 with some error content"
  154. return HttpResponseNotFound("Not found!. This page contains some MAGIC content")
  155. TestChoices = (
  156. ("a", "First Choice"),
  157. ("b", "Second Choice"),
  158. ("c", "Third Choice"),
  159. ("d", "Fourth Choice"),
  160. ("e", "Fifth Choice"),
  161. )
  162. class TestForm(Form):
  163. text = fields.CharField()
  164. email = fields.EmailField()
  165. value = fields.IntegerField()
  166. single = fields.ChoiceField(choices=TestChoices)
  167. multi = fields.MultipleChoiceField(choices=TestChoices)
  168. def clean(self):
  169. cleaned_data = self.cleaned_data
  170. if cleaned_data.get("text") == "Raise non-field error":
  171. raise ValidationError("Non-field error.")
  172. return cleaned_data
  173. def form_view(request):
  174. "A view that tests a simple form"
  175. if request.method == "POST":
  176. form = TestForm(request.POST)
  177. if form.is_valid():
  178. t = Template("Valid POST data.", name="Valid POST Template")
  179. c = Context()
  180. else:
  181. t = Template(
  182. "Invalid POST data. {{ form.errors }}", name="Invalid POST Template"
  183. )
  184. c = Context({"form": form})
  185. else:
  186. form = TestForm(request.GET)
  187. t = Template("Viewing base form. {{ form }}.", name="Form GET Template")
  188. c = Context({"form": form})
  189. return HttpResponse(t.render(c))
  190. def form_view_with_template(request):
  191. "A view that tests a simple form"
  192. if request.method == "POST":
  193. form = TestForm(request.POST)
  194. if form.is_valid():
  195. message = "POST data OK"
  196. else:
  197. message = "POST data has errors"
  198. else:
  199. form = TestForm()
  200. message = "GET form page"
  201. return render(
  202. request,
  203. "form_view.html",
  204. {
  205. "form": form,
  206. "message": message,
  207. },
  208. )
  209. @login_required
  210. def login_protected_view(request):
  211. "A simple view that is login protected."
  212. t = Template(
  213. "This is a login protected test. Username is {{ user.username }}.",
  214. name="Login Template",
  215. )
  216. c = Context({"user": request.user})
  217. return HttpResponse(t.render(c))
  218. @login_required(redirect_field_name="redirect_to")
  219. def login_protected_view_changed_redirect(request):
  220. "A simple view that is login protected with a custom redirect field set"
  221. t = Template(
  222. "This is a login protected test. Username is {{ user.username }}.",
  223. name="Login Template",
  224. )
  225. c = Context({"user": request.user})
  226. return HttpResponse(t.render(c))
  227. def _permission_protected_view(request):
  228. "A simple view that is permission protected."
  229. t = Template(
  230. "This is a permission protected test. "
  231. "Username is {{ user.username }}. "
  232. "Permissions are {{ user.get_all_permissions }}.",
  233. name="Permissions Template",
  234. )
  235. c = Context({"user": request.user})
  236. return HttpResponse(t.render(c))
  237. permission_protected_view = permission_required("permission_not_granted")(
  238. _permission_protected_view
  239. )
  240. permission_protected_view_exception = permission_required(
  241. "permission_not_granted", raise_exception=True
  242. )(_permission_protected_view)
  243. class _ViewManager:
  244. @method_decorator(login_required)
  245. def login_protected_view(self, request):
  246. t = Template(
  247. "This is a login protected test using a method. "
  248. "Username is {{ user.username }}.",
  249. name="Login Method Template",
  250. )
  251. c = Context({"user": request.user})
  252. return HttpResponse(t.render(c))
  253. @method_decorator(permission_required("permission_not_granted"))
  254. def permission_protected_view(self, request):
  255. t = Template(
  256. "This is a permission protected test using a method. "
  257. "Username is {{ user.username }}. "
  258. "Permissions are {{ user.get_all_permissions }}.",
  259. name="Permissions Template",
  260. )
  261. c = Context({"user": request.user})
  262. return HttpResponse(t.render(c))
  263. _view_manager = _ViewManager()
  264. login_protected_method_view = _view_manager.login_protected_view
  265. permission_protected_method_view = _view_manager.permission_protected_view
  266. def session_view(request):
  267. "A view that modifies the session"
  268. request.session["tobacconist"] = "hovercraft"
  269. t = Template(
  270. "This is a view that modifies the session.",
  271. name="Session Modifying View Template",
  272. )
  273. c = Context()
  274. return HttpResponse(t.render(c))
  275. def broken_view(request):
  276. """A view which just raises an exception, simulating a broken view."""
  277. raise KeyError("Oops! Looks like you wrote some bad code.")
  278. def mail_sending_view(request):
  279. mail.EmailMessage(
  280. "Test message",
  281. "This is a test email",
  282. "from@example.com",
  283. ["first@example.com", "second@example.com"],
  284. ).send()
  285. return HttpResponse("Mail sent")
  286. def mass_mail_sending_view(request):
  287. m1 = mail.EmailMessage(
  288. "First Test message",
  289. "This is the first test email",
  290. "from@example.com",
  291. ["first@example.com", "second@example.com"],
  292. )
  293. m2 = mail.EmailMessage(
  294. "Second Test message",
  295. "This is the second test email",
  296. "from@example.com",
  297. ["second@example.com", "third@example.com"],
  298. )
  299. c = mail.get_connection()
  300. c.send_messages([m1, m2])
  301. return HttpResponse("Mail sent")
  302. def nesting_exception_view(request):
  303. """
  304. A view that uses a nested client to call another view and then raises an
  305. exception.
  306. """
  307. client = Client()
  308. client.get("/get_view/")
  309. raise Exception("exception message")
  310. def django_project_redirect(request):
  311. return HttpResponseRedirect("https://www.djangoproject.com/")
  312. def no_trailing_slash_external_redirect(request):
  313. """
  314. RFC 3986 Section 6.2.3: Empty path should be normalized to "/".
  315. Use https://testserver, rather than an external domain, in order to allow
  316. use of follow=True, triggering Client._handle_redirects().
  317. """
  318. return HttpResponseRedirect("https://testserver")
  319. def index_view(request):
  320. """Target for no_trailing_slash_external_redirect with follow=True."""
  321. return HttpResponse("Hello world")
  322. def upload_view(request):
  323. """Prints keys of request.FILES to the response."""
  324. return HttpResponse(", ".join(request.FILES))
  325. class TwoArgException(Exception):
  326. def __init__(self, one, two):
  327. pass
  328. def two_arg_exception(request):
  329. raise TwoArgException("one", "two")
  330. class CBView(TemplateView):
  331. template_name = "base.html"