1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501 |
- # coding: utf-8
- from __future__ import unicode_literals
- import os
- import re
- import datetime
- import unittest
- try:
- from urllib.parse import urljoin
- except ImportError: # Python 2
- from urlparse import urljoin
- from django.conf import settings, global_settings
- from django.core import mail
- from django.core.files import temp as tempfile
- from django.core.urlresolvers import reverse
- # Register auth models with the admin.
- from django.contrib.auth import get_permission_codename
- from django.contrib.admin.helpers import ACTION_CHECKBOX_NAME
- from django.contrib.admin.models import LogEntry, DELETION
- from django.contrib.admin.sites import LOGIN_FORM_KEY
- from django.contrib.admin.util import quote
- from django.contrib.admin.views.main import IS_POPUP_VAR
- from django.contrib.admin.tests import AdminSeleniumWebDriverTestCase
- from django.contrib.auth import REDIRECT_FIELD_NAME
- from django.contrib.auth.models import Group, User, Permission
- from django.contrib.contenttypes.models import ContentType
- from django.db import connection
- from django.forms.util import ErrorList
- from django.template.response import TemplateResponse
- from django.test import TestCase
- from django.test.utils import patch_logger
- from django.utils import formats
- from django.utils import translation
- from django.utils.cache import get_max_age
- from django.utils.encoding import iri_to_uri, force_bytes
- from django.utils.html import escape
- from django.utils.http import urlencode, urlquote
- from django.utils._os import upath
- from django.utils import six
- from django.test.utils import override_settings
- # local test models
- from .models import (Article, BarAccount, CustomArticle, EmptyModel, FooAccount,
- Gallery, ModelWithStringPrimaryKey, Person, Persona, Picture, Podcast,
- Section, Subscriber, Vodcast, Language, Collector, Widget, Grommet,
- DooHickey, FancyDoodad, Whatsit, Category, Post, Plot, FunkyTag, Chapter,
- Book, Promo, WorkHour, Employee, Question, Answer, Inquisition, Actor,
- FoodDelivery, RowLevelChangePermissionModel, Paper, CoverLetter, Story,
- OtherStory, ComplexSortedPerson, PluggableSearchPerson, Parent, Child, AdminOrderedField,
- AdminOrderedModelMethod, AdminOrderedAdminMethod, AdminOrderedCallable,
- Report, MainPrepopulated, RelatedPrepopulated, UnorderedObject,
- Simple, UndeletableObject, UnchangeableObject, Choice, ShortMessage, Telegram)
- from .admin import site, site2
- ERROR_MESSAGE = "Please enter the correct username and password \
- for a staff account. Note that both fields may be case-sensitive."
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminViewBasicTestCase(TestCase):
- fixtures = ['admin-views-users.xml', 'admin-views-colors.xml',
- 'admin-views-fabrics.xml', 'admin-views-books.xml']
- # Store the bit of the URL where the admin is registered as a class
- # variable. That way we can test a second AdminSite just by subclassing
- # this test case and changing urlbit.
- urlbit = 'admin'
- urls = "admin_views.urls"
- def setUp(self):
- self.old_USE_I18N = settings.USE_I18N
- self.old_USE_L10N = settings.USE_L10N
- self.old_LANGUAGE_CODE = settings.LANGUAGE_CODE
- self.client.login(username='super', password='secret')
- settings.USE_I18N = True
- def tearDown(self):
- settings.USE_I18N = self.old_USE_I18N
- settings.USE_L10N = self.old_USE_L10N
- settings.LANGUAGE_CODE = self.old_LANGUAGE_CODE
- self.client.logout()
- formats.reset_format_cache()
- def assertContentBefore(self, response, text1, text2, failing_msg=None):
- """
- Testing utility asserting that text1 appears before text2 in response
- content.
- """
- self.assertEqual(response.status_code, 200)
- self.assertTrue(response.content.index(force_bytes(text1)) < response.content.index(force_bytes(text2)),
- failing_msg
- )
- class AdminViewBasicTest(AdminViewBasicTestCase):
- def testTrailingSlashRequired(self):
- """
- If you leave off the trailing slash, app should redirect and add it.
- """
- response = self.client.get('/test_admin/%s/admin_views/article/add' % self.urlbit)
- self.assertRedirects(response,
- '/test_admin/%s/admin_views/article/add/' % self.urlbit, status_code=301
- )
- def testBasicAddGet(self):
- """
- A smoke test to ensure GET on the add_view works.
- """
- response = self.client.get('/test_admin/%s/admin_views/section/add/' % self.urlbit)
- self.assertIsInstance(response, TemplateResponse)
- self.assertEqual(response.status_code, 200)
- def testAddWithGETArgs(self):
- response = self.client.get('/test_admin/%s/admin_views/section/add/' % self.urlbit, {'name': 'My Section'})
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'value="My Section"',
- msg_prefix="Couldn't find an input with the right value in the response"
- )
- def testBasicEditGet(self):
- """
- A smoke test to ensure GET on the change_view works.
- """
- response = self.client.get('/test_admin/%s/admin_views/section/1/' % self.urlbit)
- self.assertIsInstance(response, TemplateResponse)
- self.assertEqual(response.status_code, 200)
- def testBasicEditGetStringPK(self):
- """
- Ensure GET on the change_view works (returns an HTTP 404 error, see
- #11191) when passing a string as the PK argument for a model with an
- integer PK field.
- """
- response = self.client.get('/test_admin/%s/admin_views/section/abc/' % self.urlbit)
- self.assertEqual(response.status_code, 404)
- def testBasicInheritanceGetStringPK(self):
- """
- Ensure GET on the change_view works on inherited models (returns an
- HTTP 404 error, see #19951) when passing a string as the PK argument
- for a model with an integer PK field.
- """
- response = self.client.get('/test_admin/%s/admin_views/supervillain/abc/' % self.urlbit)
- self.assertEqual(response.status_code, 404)
- def testBasicAddPost(self):
- """
- A smoke test to ensure POST on add_view works.
- """
- post_data = {
- "name": "Another Section",
- # inline data
- "article_set-TOTAL_FORMS": "3",
- "article_set-INITIAL_FORMS": "0",
- "article_set-MAX_NUM_FORMS": "0",
- }
- response = self.client.post('/test_admin/%s/admin_views/section/add/' % self.urlbit, post_data)
- self.assertEqual(response.status_code, 302) # redirect somewhere
- def testPopupAddPost(self):
- """
- Ensure http response from a popup is properly escaped.
- """
- post_data = {
- '_popup': '1',
- 'title': 'title with a new\nline',
- 'content': 'some content',
- 'date_0': '2010-09-10',
- 'date_1': '14:55:39',
- }
- response = self.client.post('/test_admin/%s/admin_views/article/add/' % self.urlbit, post_data)
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'dismissAddAnotherPopup')
- self.assertContains(response, 'title with a new\\u000Aline')
- # Post data for edit inline
- inline_post_data = {
- "name": "Test section",
- # inline data
- "article_set-TOTAL_FORMS": "6",
- "article_set-INITIAL_FORMS": "3",
- "article_set-MAX_NUM_FORMS": "0",
- "article_set-0-id": "1",
- # there is no title in database, give one here or formset will fail.
- "article_set-0-title": "Norske bostaver æøå skaper problemer",
- "article_set-0-content": "<p>Middle content</p>",
- "article_set-0-date_0": "2008-03-18",
- "article_set-0-date_1": "11:54:58",
- "article_set-0-section": "1",
- "article_set-1-id": "2",
- "article_set-1-title": "Need a title.",
- "article_set-1-content": "<p>Oldest content</p>",
- "article_set-1-date_0": "2000-03-18",
- "article_set-1-date_1": "11:54:58",
- "article_set-2-id": "3",
- "article_set-2-title": "Need a title.",
- "article_set-2-content": "<p>Newest content</p>",
- "article_set-2-date_0": "2009-03-18",
- "article_set-2-date_1": "11:54:58",
- "article_set-3-id": "",
- "article_set-3-title": "",
- "article_set-3-content": "",
- "article_set-3-date_0": "",
- "article_set-3-date_1": "",
- "article_set-4-id": "",
- "article_set-4-title": "",
- "article_set-4-content": "",
- "article_set-4-date_0": "",
- "article_set-4-date_1": "",
- "article_set-5-id": "",
- "article_set-5-title": "",
- "article_set-5-content": "",
- "article_set-5-date_0": "",
- "article_set-5-date_1": "",
- }
- def testBasicEditPost(self):
- """
- A smoke test to ensure POST on edit_view works.
- """
- response = self.client.post('/test_admin/%s/admin_views/section/1/' % self.urlbit, self.inline_post_data)
- self.assertEqual(response.status_code, 302) # redirect somewhere
- def testEditSaveAs(self):
- """
- Test "save as".
- """
- post_data = self.inline_post_data.copy()
- post_data.update({
- '_saveasnew': 'Save+as+new',
- "article_set-1-section": "1",
- "article_set-2-section": "1",
- "article_set-3-section": "1",
- "article_set-4-section": "1",
- "article_set-5-section": "1",
- })
- response = self.client.post('/test_admin/%s/admin_views/section/1/' % self.urlbit, post_data)
- self.assertEqual(response.status_code, 302) # redirect somewhere
- def testChangeListSortingCallable(self):
- """
- Ensure we can sort on a list_display field that is a callable
- (column 2 is callable_year in ArticleAdmin)
- """
- response = self.client.get('/test_admin/%s/admin_views/article/' % self.urlbit, {'o': 2})
- self.assertContentBefore(response, 'Oldest content', 'Middle content',
- "Results of sorting on callable are out of order.")
- self.assertContentBefore(response, 'Middle content', 'Newest content',
- "Results of sorting on callable are out of order.")
- def testChangeListSortingModel(self):
- """
- Ensure we can sort on a list_display field that is a Model method
- (colunn 3 is 'model_year' in ArticleAdmin)
- """
- response = self.client.get('/test_admin/%s/admin_views/article/' % self.urlbit, {'o': '-3'})
- self.assertContentBefore(response, 'Newest content', 'Middle content',
- "Results of sorting on Model method are out of order.")
- self.assertContentBefore(response, 'Middle content', 'Oldest content',
- "Results of sorting on Model method are out of order.")
- def testChangeListSortingModelAdmin(self):
- """
- Ensure we can sort on a list_display field that is a ModelAdmin method
- (colunn 4 is 'modeladmin_year' in ArticleAdmin)
- """
- response = self.client.get('/test_admin/%s/admin_views/article/' % self.urlbit, {'o': '4'})
- self.assertContentBefore(response, 'Oldest content', 'Middle content',
- "Results of sorting on ModelAdmin method are out of order.")
- self.assertContentBefore(response, 'Middle content', 'Newest content',
- "Results of sorting on ModelAdmin method are out of order.")
- def testChangeListSortingMultiple(self):
- p1 = Person.objects.create(name="Chris", gender=1, alive=True)
- p2 = Person.objects.create(name="Chris", gender=2, alive=True)
- p3 = Person.objects.create(name="Bob", gender=1, alive=True)
- link1 = reverse('admin:admin_views_person_change', args=(p1.pk,))
- link2 = reverse('admin:admin_views_person_change', args=(p2.pk,))
- link3 = reverse('admin:admin_views_person_change', args=(p3.pk,))
- # Sort by name, gender
- # This hard-codes the URL because it'll fail if it runs against the
- # 'admin2' custom admin (which doesn't have the Person model).
- response = self.client.get('/test_admin/admin/admin_views/person/', {'o': '1.2'})
- self.assertContentBefore(response, link3, link1)
- self.assertContentBefore(response, link1, link2)
- # Sort by gender descending, name
- response = self.client.get('/test_admin/admin/admin_views/person/', {'o': '-2.1'})
- self.assertContentBefore(response, link2, link3)
- self.assertContentBefore(response, link3, link1)
- def testChangeListSortingPreserveQuerySetOrdering(self):
- """
- If no ordering is defined in `ModelAdmin.ordering` or in the query
- string, then the underlying order of the queryset should not be
- changed, even if it is defined in `Modeladmin.get_queryset()`.
- Refs #11868, #7309.
- """
- p1 = Person.objects.create(name="Amy", gender=1, alive=True, age=80)
- p2 = Person.objects.create(name="Bob", gender=1, alive=True, age=70)
- p3 = Person.objects.create(name="Chris", gender=2, alive=False, age=60)
- link1 = reverse('admin:admin_views_person_change', args=(p1.pk,))
- link2 = reverse('admin:admin_views_person_change', args=(p2.pk,))
- link3 = reverse('admin:admin_views_person_change', args=(p3.pk,))
- # This hard-codes the URL because it'll fail if it runs against the
- # 'admin2' custom admin (which doesn't have the Person model).
- response = self.client.get('/test_admin/admin/admin_views/person/', {})
- self.assertContentBefore(response, link3, link2)
- self.assertContentBefore(response, link2, link1)
- def testChangeListSortingModelMeta(self):
- # Test ordering on Model Meta is respected
- l1 = Language.objects.create(iso='ur', name='Urdu')
- l2 = Language.objects.create(iso='ar', name='Arabic')
- link1 = reverse('admin:admin_views_language_change', args=(quote(l1.pk),))
- link2 = reverse('admin:admin_views_language_change', args=(quote(l2.pk),))
- response = self.client.get('/test_admin/admin/admin_views/language/', {})
- self.assertContentBefore(response, link2, link1)
- # Test we can override with query string
- response = self.client.get('/test_admin/admin/admin_views/language/', {'o': '-1'})
- self.assertContentBefore(response, link1, link2)
- def testChangeListSortingOverrideModelAdmin(self):
- # Test ordering on Model Admin is respected, and overrides Model Meta
- dt = datetime.datetime.now()
- p1 = Podcast.objects.create(name="A", release_date=dt)
- p2 = Podcast.objects.create(name="B", release_date=dt - datetime.timedelta(10))
- link1 = reverse('admin:admin_views_podcast_change', args=(p1.pk,))
- link2 = reverse('admin:admin_views_podcast_change', args=(p2.pk,))
- response = self.client.get('/test_admin/admin/admin_views/podcast/', {})
- self.assertContentBefore(response, link1, link2)
- def testMultipleSortSameField(self):
- # Check that we get the columns we expect if we have two columns
- # that correspond to the same ordering field
- dt = datetime.datetime.now()
- p1 = Podcast.objects.create(name="A", release_date=dt)
- p2 = Podcast.objects.create(name="B", release_date=dt - datetime.timedelta(10))
- link1 = reverse('admin:admin_views_podcast_change', args=(quote(p1.pk),))
- link2 = reverse('admin:admin_views_podcast_change', args=(quote(p2.pk),))
- response = self.client.get('/test_admin/admin/admin_views/podcast/', {})
- self.assertContentBefore(response, link1, link2)
- p1 = ComplexSortedPerson.objects.create(name="Bob", age=10)
- p2 = ComplexSortedPerson.objects.create(name="Amy", age=20)
- link1 = reverse('admin:admin_views_complexsortedperson_change', args=(p1.pk,))
- link2 = reverse('admin:admin_views_complexsortedperson_change', args=(p2.pk,))
- response = self.client.get('/test_admin/admin/admin_views/complexsortedperson/', {})
- # Should have 5 columns (including action checkbox col)
- self.assertContains(response, '<th scope="col"', count=5)
- self.assertContains(response, 'Name')
- self.assertContains(response, 'Colored name')
- # Check order
- self.assertContentBefore(response, 'Name', 'Colored name')
- # Check sorting - should be by name
- self.assertContentBefore(response, link2, link1)
- def testSortIndicatorsAdminOrder(self):
- """
- Ensures that the admin shows default sort indicators for all
- kinds of 'ordering' fields: field names, method on the model
- admin and model itself, and other callables. See #17252.
- """
- models = [(AdminOrderedField, 'adminorderedfield'),
- (AdminOrderedModelMethod, 'adminorderedmodelmethod'),
- (AdminOrderedAdminMethod, 'adminorderedadminmethod'),
- (AdminOrderedCallable, 'adminorderedcallable')]
- for model, url in models:
- a1 = model.objects.create(stuff='The Last Item', order=3)
- a2 = model.objects.create(stuff='The First Item', order=1)
- a3 = model.objects.create(stuff='The Middle Item', order=2)
- response = self.client.get('/test_admin/admin/admin_views/%s/' % url, {})
- self.assertEqual(response.status_code, 200)
- # Should have 3 columns including action checkbox col.
- self.assertContains(response, '<th scope="col"', count=3, msg_prefix=url)
- # Check if the correct column was selected. 2 is the index of the
- # 'order' column in the model admin's 'list_display' with 0 being
- # the implicit 'action_checkbox' and 1 being the column 'stuff'.
- self.assertEqual(response.context['cl'].get_ordering_field_columns(), {2: 'asc'})
- # Check order of records.
- self.assertContentBefore(response, 'The First Item', 'The Middle Item')
- self.assertContentBefore(response, 'The Middle Item', 'The Last Item')
- def testLimitedFilter(self):
- """Ensure admin changelist filters do not contain objects excluded via limit_choices_to.
- This also tests relation-spanning filters (e.g. 'color__value').
- """
- response = self.client.get('/test_admin/%s/admin_views/thing/' % self.urlbit)
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, '<div id="changelist-filter">',
- msg_prefix="Expected filter not found in changelist view"
- )
- self.assertNotContains(response, '<a href="?color__id__exact=3">Blue</a>',
- msg_prefix="Changelist filter not correctly limited by limit_choices_to"
- )
- def testRelationSpanningFilters(self):
- response = self.client.get('/test_admin/%s/admin_views/chapterxtra1/' %
- self.urlbit)
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, '<div id="changelist-filter">')
- filters = {
- 'chap__id__exact': dict(
- values=[c.id for c in Chapter.objects.all()],
- test=lambda obj, value: obj.chap.id == value),
- 'chap__title': dict(
- values=[c.title for c in Chapter.objects.all()],
- test=lambda obj, value: obj.chap.title == value),
- 'chap__book__id__exact': dict(
- values=[b.id for b in Book.objects.all()],
- test=lambda obj, value: obj.chap.book.id == value),
- 'chap__book__name': dict(
- values=[b.name for b in Book.objects.all()],
- test=lambda obj, value: obj.chap.book.name == value),
- 'chap__book__promo__id__exact': dict(
- values=[p.id for p in Promo.objects.all()],
- test=lambda obj, value:
- obj.chap.book.promo_set.filter(id=value).exists()),
- 'chap__book__promo__name': dict(
- values=[p.name for p in Promo.objects.all()],
- test=lambda obj, value:
- obj.chap.book.promo_set.filter(name=value).exists()),
- }
- for filter_path, params in filters.items():
- for value in params['values']:
- query_string = urlencode({filter_path: value})
- # ensure filter link exists
- self.assertContains(response, '<a href="?%s">' % query_string)
- # ensure link works
- filtered_response = self.client.get(
- '/test_admin/%s/admin_views/chapterxtra1/?%s' % (
- self.urlbit, query_string))
- self.assertEqual(filtered_response.status_code, 200)
- # ensure changelist contains only valid objects
- for obj in filtered_response.context['cl'].queryset.all():
- self.assertTrue(params['test'](obj, value))
- def testIncorrectLookupParameters(self):
- """Ensure incorrect lookup parameters are handled gracefully."""
- response = self.client.get('/test_admin/%s/admin_views/thing/' % self.urlbit, {'notarealfield': '5'})
- self.assertRedirects(response, '/test_admin/%s/admin_views/thing/?e=1' % self.urlbit)
- # Spanning relationships through an inexistant related object (Refs #16716)
- response = self.client.get('/test_admin/%s/admin_views/thing/' % self.urlbit, {'notarealfield__whatever': '5'})
- self.assertRedirects(response, '/test_admin/%s/admin_views/thing/?e=1' % self.urlbit)
- response = self.client.get('/test_admin/%s/admin_views/thing/' % self.urlbit, {'color__id__exact': 'StringNotInteger!'})
- self.assertRedirects(response, '/test_admin/%s/admin_views/thing/?e=1' % self.urlbit)
- # Regression test for #18530
- response = self.client.get('/test_admin/%s/admin_views/thing/' % self.urlbit, {'pub_date__gte': 'foo'})
- self.assertRedirects(response, '/test_admin/%s/admin_views/thing/?e=1' % self.urlbit)
- def testIsNullLookups(self):
- """Ensure is_null is handled correctly."""
- Article.objects.create(title="I Could Go Anywhere", content="Versatile", date=datetime.datetime.now())
- response = self.client.get('/test_admin/%s/admin_views/article/' % self.urlbit)
- self.assertContains(response, '4 articles')
- response = self.client.get('/test_admin/%s/admin_views/article/' % self.urlbit, {'section__isnull': 'false'})
- self.assertContains(response, '3 articles')
- response = self.client.get('/test_admin/%s/admin_views/article/' % self.urlbit, {'section__isnull': '0'})
- self.assertContains(response, '3 articles')
- response = self.client.get('/test_admin/%s/admin_views/article/' % self.urlbit, {'section__isnull': 'true'})
- self.assertContains(response, '1 article')
- response = self.client.get('/test_admin/%s/admin_views/article/' % self.urlbit, {'section__isnull': '1'})
- self.assertContains(response, '1 article')
- def testLogoutAndPasswordChangeURLs(self):
- response = self.client.get('/test_admin/%s/admin_views/article/' % self.urlbit)
- self.assertContains(response, '<a href="/test_admin/%s/logout/">' % self.urlbit)
- self.assertContains(response, '<a href="/test_admin/%s/password_change/">' % self.urlbit)
- def testNamedGroupFieldChoicesChangeList(self):
- """
- Ensures the admin changelist shows correct values in the relevant column
- for rows corresponding to instances of a model in which a named group
- has been used in the choices option of a field.
- """
- link1 = reverse('admin:admin_views_fabric_change', args=(1,), current_app=self.urlbit)
- link2 = reverse('admin:admin_views_fabric_change', args=(2,), current_app=self.urlbit)
- response = self.client.get('/test_admin/%s/admin_views/fabric/' % self.urlbit)
- fail_msg = "Changelist table isn't showing the right human-readable values set by a model field 'choices' option named group."
- self.assertContains(response, '<a href="%s">Horizontal</a>' % link1, msg_prefix=fail_msg, html=True)
- self.assertContains(response, '<a href="%s">Vertical</a>' % link2, msg_prefix=fail_msg, html=True)
- def testNamedGroupFieldChoicesFilter(self):
- """
- Ensures the filter UI shows correctly when at least one named group has
- been used in the choices option of a model field.
- """
- response = self.client.get('/test_admin/%s/admin_views/fabric/' % self.urlbit)
- fail_msg = "Changelist filter isn't showing options contained inside a model field 'choices' option named group."
- self.assertContains(response, '<div id="changelist-filter">')
- self.assertContains(response,
- '<a href="?surface__exact=x">Horizontal</a>', msg_prefix=fail_msg, html=True)
- self.assertContains(response,
- '<a href="?surface__exact=y">Vertical</a>', msg_prefix=fail_msg, html=True)
- def testChangeListNullBooleanDisplay(self):
- Post.objects.create(public=None)
- # This hard-codes the URl because it'll fail if it runs
- # against the 'admin2' custom admin (which doesn't have the
- # Post model).
- response = self.client.get("/test_admin/admin/admin_views/post/")
- self.assertContains(response, 'icon-unknown.gif')
- def testI18NLanguageNonEnglishDefault(self):
- """
- Check if the JavaScript i18n view returns an empty language catalog
- if the default language is non-English but the selected language
- is English. See #13388 and #3594 for more details.
- """
- with self.settings(LANGUAGE_CODE='fr'), translation.override('en-us'):
- response = self.client.get('/test_admin/admin/jsi18n/')
- self.assertNotContains(response, 'Choisir une heure')
- def testI18NLanguageNonEnglishFallback(self):
- """
- Makes sure that the fallback language is still working properly
- in cases where the selected language cannot be found.
- """
- with self.settings(LANGUAGE_CODE='fr'), translation.override('none'):
- response = self.client.get('/test_admin/admin/jsi18n/')
- self.assertContains(response, 'Choisir une heure')
- def testL10NDeactivated(self):
- """
- Check if L10N is deactivated, the JavaScript i18n view doesn't
- return localized date/time formats. Refs #14824.
- """
- with self.settings(LANGUAGE_CODE='ru', USE_L10N=False), translation.override('none'):
- response = self.client.get('/test_admin/admin/jsi18n/')
- self.assertNotContains(response, '%d.%m.%Y %H:%M:%S')
- self.assertContains(response, '%Y-%m-%d %H:%M:%S')
- def test_disallowed_filtering(self):
- with patch_logger('django.security.DisallowedModelAdminLookup', 'error') as calls:
- response = self.client.get("/test_admin/admin/admin_views/album/?owner__email__startswith=fuzzy")
- self.assertEqual(response.status_code, 400)
- self.assertEqual(len(calls), 1)
- # Filters are allowed if explicitly included in list_filter
- response = self.client.get("/test_admin/admin/admin_views/thing/?color__value__startswith=red")
- self.assertEqual(response.status_code, 200)
- response = self.client.get("/test_admin/admin/admin_views/thing/?color__value=red")
- self.assertEqual(response.status_code, 200)
- # Filters should be allowed if they involve a local field without the
- # need to whitelist them in list_filter or date_hierarchy.
- response = self.client.get("/test_admin/admin/admin_views/person/?age__gt=30")
- self.assertEqual(response.status_code, 200)
- e1 = Employee.objects.create(name='Anonymous', gender=1, age=22, alive=True, code='123')
- e2 = Employee.objects.create(name='Visitor', gender=2, age=19, alive=True, code='124')
- WorkHour.objects.create(datum=datetime.datetime.now(), employee=e1)
- WorkHour.objects.create(datum=datetime.datetime.now(), employee=e2)
- response = self.client.get("/test_admin/admin/admin_views/workhour/")
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'employee__person_ptr__exact')
- response = self.client.get("/test_admin/admin/admin_views/workhour/?employee__person_ptr__exact=%d" % e1.pk)
- self.assertEqual(response.status_code, 200)
- def test_allowed_filtering_15103(self):
- """
- Regressions test for ticket 15103 - filtering on fields defined in a
- ForeignKey 'limit_choices_to' should be allowed, otherwise raw_id_fields
- can break.
- """
- # Filters should be allowed if they are defined on a ForeignKey pointing to this model
- response = self.client.get("/test_admin/admin/admin_views/inquisition/?leader__name=Palin&leader__age=27")
- self.assertEqual(response.status_code, 200)
- def test_popup_dismiss_related(self):
- """
- Regression test for ticket 20664 - ensure the pk is properly quoted.
- """
- actor = Actor.objects.create(name="Palin", age=27)
- response = self.client.get("/test_admin/admin/admin_views/actor/?%s" % IS_POPUP_VAR)
- self.assertContains(response, "opener.dismissRelatedLookupPopup(window, '%s')" % actor.pk)
- def test_hide_change_password(self):
- """
- Tests if the "change password" link in the admin is hidden if the User
- does not have a usable password set.
- (against 9bea85795705d015cdadc82c68b99196a8554f5c)
- """
- user = User.objects.get(username='super')
- password = user.password
- user.set_unusable_password()
- user.save()
- response = self.client.get('/test_admin/admin/')
- self.assertNotContains(response, reverse('admin:password_change'),
- msg_prefix='The "change password" link should not be displayed if a user does not have a usable password.')
- def test_change_view_with_show_delete_extra_context(self):
- """
- Ensured that the 'show_delete' context variable in the admin's change
- view actually controls the display of the delete button.
- Refs #10057.
- """
- instance = UndeletableObject.objects.create(name='foo')
- response = self.client.get('/test_admin/%s/admin_views/undeletableobject/%d/' %
- (self.urlbit, instance.pk))
- self.assertNotContains(response, 'deletelink')
- def test_allows_attributeerror_to_bubble_up(self):
- """
- Ensure that AttributeErrors are allowed to bubble when raised inside
- a change list view.
- Requires a model to be created so there's something to be displayed
- Refs: #16655, #18593, and #18747
- """
- Simple.objects.create()
- with self.assertRaises(AttributeError):
- self.client.get('/test_admin/%s/admin_views/simple/' % self.urlbit)
- def test_changelist_with_no_change_url(self):
- """
- ModelAdmin.changelist_view shouldn't result in a NoReverseMatch if url
- for change_view is removed from get_urls
- Regression test for #20934
- """
- UnchangeableObject.objects.create()
- response = self.client.get('/test_admin/admin/admin_views/unchangeableobject/')
- self.assertEqual(response.status_code, 200)
- # Check the format of the shown object -- shouldn't contain a change link
- self.assertContains(response, '<th class="field-__str__">UnchangeableObject object</th>', html=True)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminViewFormUrlTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ["admin-views-users.xml"]
- urlbit = "admin3"
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def testChangeFormUrlHasCorrectValue(self):
- """
- Tests whether change_view has form_url in response.context
- """
- response = self.client.get('/test_admin/%s/admin_views/section/1/' % self.urlbit)
- self.assertTrue('form_url' in response.context, msg='form_url not present in response.context')
- self.assertEqual(response.context['form_url'], 'pony')
- def test_filter_with_custom_template(self):
- """
- Ensure that one can use a custom template to render an admin filter.
- Refs #17515.
- """
- template_dirs = settings.TEMPLATE_DIRS + (
- os.path.join(os.path.dirname(upath(__file__)), 'templates'),)
- with self.settings(TEMPLATE_DIRS=template_dirs):
- response = self.client.get("/test_admin/admin/admin_views/color2/")
- self.assertTemplateUsed(response, 'custom_filter_template.html')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminJavaScriptTest(TestCase):
- fixtures = ['admin-views-users.xml']
- urls = "admin_views.urls"
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def testSingleWidgetFirsFieldFocus(self):
- """
- JavaScript-assisted auto-focus on first field.
- """
- response = self.client.get('/test_admin/%s/admin_views/picture/add/' % 'admin')
- self.assertContains(
- response,
- '<script type="text/javascript">document.getElementById("id_name").focus();</script>'
- )
- def testMultiWidgetFirsFieldFocus(self):
- """
- JavaScript-assisted auto-focus should work if a model/ModelAdmin setup
- is such that the first form field has a MultiWidget.
- """
- response = self.client.get('/test_admin/%s/admin_views/reservation/add/' % 'admin')
- self.assertContains(
- response,
- '<script type="text/javascript">document.getElementById("id_start_date_0").focus();</script>'
- )
- def test_js_minified_only_if_debug_is_false(self):
- """
- Ensure that the minified versions of the JS files are only used when
- DEBUG is False.
- Refs #17521.
- """
- with override_settings(DEBUG=False):
- response = self.client.get(
- '/test_admin/%s/admin_views/section/add/' % 'admin')
- self.assertNotContains(response, 'jquery.js')
- self.assertContains(response, 'jquery.min.js')
- self.assertNotContains(response, 'prepopulate.js')
- self.assertContains(response, 'prepopulate.min.js')
- self.assertNotContains(response, 'actions.js')
- self.assertContains(response, 'actions.min.js')
- self.assertNotContains(response, 'collapse.js')
- self.assertContains(response, 'collapse.min.js')
- self.assertNotContains(response, 'inlines.js')
- self.assertContains(response, 'inlines.min.js')
- with override_settings(DEBUG=True):
- response = self.client.get(
- '/test_admin/%s/admin_views/section/add/' % 'admin')
- self.assertContains(response, 'jquery.js')
- self.assertNotContains(response, 'jquery.min.js')
- self.assertContains(response, 'prepopulate.js')
- self.assertNotContains(response, 'prepopulate.min.js')
- self.assertContains(response, 'actions.js')
- self.assertNotContains(response, 'actions.min.js')
- self.assertContains(response, 'collapse.js')
- self.assertNotContains(response, 'collapse.min.js')
- self.assertContains(response, 'inlines.js')
- self.assertNotContains(response, 'inlines.min.js')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class SaveAsTests(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml', 'admin-views-person.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_save_as_duplication(self):
- """Ensure save as actually creates a new person"""
- post_data = {'_saveasnew': '', 'name': 'John M', 'gender': 1, 'age': 42}
- response = self.client.post('/test_admin/admin/admin_views/person/1/', post_data)
- self.assertEqual(len(Person.objects.filter(name='John M')), 1)
- self.assertEqual(len(Person.objects.filter(id=1)), 1)
- def test_save_as_display(self):
- """
- Ensure that 'save as' is displayed when activated and after submitting
- invalid data aside save_as_new will not show us a form to overwrite the
- initial model.
- """
- response = self.client.get('/test_admin/admin/admin_views/person/1/')
- self.assertTrue(response.context['save_as'])
- post_data = {'_saveasnew': '', 'name': 'John M', 'gender': 3, 'alive': 'checked'}
- response = self.client.post('/test_admin/admin/admin_views/person/1/', post_data)
- self.assertEqual(response.context['form_url'], '/test_admin/admin/admin_views/person/add/')
- class CustomModelAdminTest(AdminViewBasicTestCase):
- urls = "admin_views.urls"
- urlbit = "admin2"
- def testCustomAdminSiteLoginForm(self):
- self.client.logout()
- response = self.client.get('/test_admin/admin2/')
- self.assertIsInstance(response, TemplateResponse)
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin2/', {
- REDIRECT_FIELD_NAME: '/test_admin/admin2/',
- LOGIN_FORM_KEY: 1,
- 'username': 'customform',
- 'password': 'secret',
- })
- self.assertIsInstance(login, TemplateResponse)
- self.assertEqual(login.status_code, 200)
- self.assertContains(login, 'custom form error')
- def testCustomAdminSiteLoginTemplate(self):
- self.client.logout()
- response = self.client.get('/test_admin/admin2/')
- self.assertIsInstance(response, TemplateResponse)
- self.assertTemplateUsed(response, 'custom_admin/login.html')
- self.assertContains(response, 'Hello from a custom login template')
- def testCustomAdminSiteLogoutTemplate(self):
- response = self.client.get('/test_admin/admin2/logout/')
- self.assertIsInstance(response, TemplateResponse)
- self.assertTemplateUsed(response, 'custom_admin/logout.html')
- self.assertContains(response, 'Hello from a custom logout template')
- def testCustomAdminSiteIndexViewAndTemplate(self):
- try:
- response = self.client.get('/test_admin/admin2/')
- except TypeError:
- self.fail('AdminSite.index_template should accept a list of template paths')
- self.assertIsInstance(response, TemplateResponse)
- self.assertTemplateUsed(response, 'custom_admin/index.html')
- self.assertContains(response, 'Hello from a custom index template *bar*')
- def testCustomAdminSiteAppIndexViewandTemplate(self):
- response = self.client.get('/test_admin/admin2/admin_views/')
- self.assertIsInstance(response, TemplateResponse)
- self.assertTemplateUsed(response, 'custom_admin/app_index.html')
- self.assertContains(response, 'Hello from a custom app_index template')
- def testCustomAdminSitePasswordChangeTemplate(self):
- response = self.client.get('/test_admin/admin2/password_change/')
- self.assertIsInstance(response, TemplateResponse)
- self.assertTemplateUsed(response, 'custom_admin/password_change_form.html')
- self.assertContains(response, 'Hello from a custom password change form template')
- def testCustomAdminSitePasswordChangeDoneTemplate(self):
- response = self.client.get('/test_admin/admin2/password_change/done/')
- self.assertIsInstance(response, TemplateResponse)
- self.assertTemplateUsed(response, 'custom_admin/password_change_done.html')
- self.assertContains(response, 'Hello from a custom password change done template')
- def testCustomAdminSiteView(self):
- self.client.login(username='super', password='secret')
- response = self.client.get('/test_admin/%s/my_view/' % self.urlbit)
- self.assertEqual(response.content, b"Django is a magical pony!")
- def test_pwd_change_custom_template(self):
- self.client.login(username='super', password='secret')
- su = User.objects.get(username='super')
- try:
- response = self.client.get('/test_admin/admin4/auth/user/%s/password/' % su.pk)
- except TypeError:
- self.fail('ModelAdmin.change_user_password_template should accept a list of template paths')
- self.assertEqual(response.status_code, 200)
- def get_perm(Model, perm):
- """Return the permission object, for the Model"""
- ct = ContentType.objects.get_for_model(Model)
- return Permission.objects.get(content_type=ct, codename=perm)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminViewPermissionsTest(TestCase):
- """Tests for Admin Views Permissions."""
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- """Test setup."""
- # Setup permissions, for our users who can add, change, and delete.
- # We can't put this into the fixture, because the content type id
- # and the permission id could be different on each run of the test.
- opts = Article._meta
- # User who can add Articles
- add_user = User.objects.get(username='adduser')
- add_user.user_permissions.add(get_perm(Article,
- get_permission_codename('add', opts)))
- # User who can change Articles
- change_user = User.objects.get(username='changeuser')
- change_user.user_permissions.add(get_perm(Article,
- get_permission_codename('change', opts)))
- # User who can delete Articles
- delete_user = User.objects.get(username='deleteuser')
- delete_user.user_permissions.add(get_perm(Article,
- get_permission_codename('delete', opts)))
- delete_user.user_permissions.add(get_perm(Section,
- get_permission_codename('delete', Section._meta)))
- # login POST dicts
- self.super_login = {
- REDIRECT_FIELD_NAME: '/test_admin/admin/',
- LOGIN_FORM_KEY: 1,
- 'username': 'super',
- 'password': 'secret',
- }
- self.super_email_login = {
- REDIRECT_FIELD_NAME: '/test_admin/admin/',
- LOGIN_FORM_KEY: 1,
- 'username': 'super@example.com',
- 'password': 'secret',
- }
- self.super_email_bad_login = {
- REDIRECT_FIELD_NAME: '/test_admin/admin/',
- LOGIN_FORM_KEY: 1,
- 'username': 'super@example.com',
- 'password': 'notsecret',
- }
- self.adduser_login = {
- REDIRECT_FIELD_NAME: '/test_admin/admin/',
- LOGIN_FORM_KEY: 1,
- 'username': 'adduser',
- 'password': 'secret',
- }
- self.changeuser_login = {
- REDIRECT_FIELD_NAME: '/test_admin/admin/',
- LOGIN_FORM_KEY: 1,
- 'username': 'changeuser',
- 'password': 'secret',
- }
- self.deleteuser_login = {
- REDIRECT_FIELD_NAME: '/test_admin/admin/',
- LOGIN_FORM_KEY: 1,
- 'username': 'deleteuser',
- 'password': 'secret',
- }
- self.joepublic_login = {
- REDIRECT_FIELD_NAME: '/test_admin/admin/',
- LOGIN_FORM_KEY: 1,
- 'username': 'joepublic',
- 'password': 'secret',
- }
- self.no_username_login = {
- REDIRECT_FIELD_NAME: '/test_admin/admin/',
- LOGIN_FORM_KEY: 1,
- 'password': 'secret',
- }
- def testLogin(self):
- """
- Make sure only staff members can log in.
- Successful posts to the login page will redirect to the orignal url.
- Unsuccessfull attempts will continue to render the login page with
- a 200 status code.
- """
- # Super User
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/', self.super_login)
- self.assertRedirects(login, '/test_admin/admin/')
- self.assertFalse(login.context)
- self.client.get('/test_admin/admin/logout/')
- # Test if user enters email address
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/', self.super_email_login)
- self.assertContains(login, ERROR_MESSAGE)
- # only correct passwords get a username hint
- login = self.client.post('/test_admin/admin/', self.super_email_bad_login)
- self.assertContains(login, ERROR_MESSAGE)
- new_user = User(username='jondoe', password='secret', email='super@example.com')
- new_user.save()
- # check to ensure if there are multiple email addresses a user doesn't get a 500
- login = self.client.post('/test_admin/admin/', self.super_email_login)
- self.assertContains(login, ERROR_MESSAGE)
- # Add User
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/', self.adduser_login)
- self.assertRedirects(login, '/test_admin/admin/')
- self.assertFalse(login.context)
- self.client.get('/test_admin/admin/logout/')
- # Change User
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/', self.changeuser_login)
- self.assertRedirects(login, '/test_admin/admin/')
- self.assertFalse(login.context)
- self.client.get('/test_admin/admin/logout/')
- # Delete User
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/', self.deleteuser_login)
- self.assertRedirects(login, '/test_admin/admin/')
- self.assertFalse(login.context)
- self.client.get('/test_admin/admin/logout/')
- # Regular User should not be able to login.
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/', self.joepublic_login)
- self.assertEqual(login.status_code, 200)
- self.assertContains(login, ERROR_MESSAGE)
- # Requests without username should not return 500 errors.
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/', self.no_username_login)
- self.assertEqual(login.status_code, 200)
- form = login.context[0].get('form')
- self.assertEqual(form.errors['username'][0], 'This field is required.')
- def testLoginSuccessfullyRedirectsToOriginalUrl(self):
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(response.status_code, 200)
- query_string = 'the-answer=42'
- redirect_url = '/test_admin/admin/?%s' % query_string
- new_next = {REDIRECT_FIELD_NAME: redirect_url}
- login = self.client.post('/test_admin/admin/', dict(self.super_login, **new_next), QUERY_STRING=query_string)
- self.assertRedirects(login, redirect_url)
- def testDoubleLoginIsNotAllowed(self):
- """Regression test for #19327"""
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(response.status_code, 200)
- # Establish a valid admin session
- login = self.client.post('/test_admin/admin/', self.super_login)
- self.assertRedirects(login, '/test_admin/admin/')
- self.assertFalse(login.context)
- # Logging in with non-admin user fails
- login = self.client.post('/test_admin/admin/', self.joepublic_login)
- self.assertEqual(login.status_code, 200)
- self.assertContains(login, ERROR_MESSAGE)
- # Establish a valid admin session
- login = self.client.post('/test_admin/admin/', self.super_login)
- self.assertRedirects(login, '/test_admin/admin/')
- self.assertFalse(login.context)
- # Logging in with admin user while already logged in
- login = self.client.post('/test_admin/admin/', self.super_login)
- self.assertRedirects(login, '/test_admin/admin/')
- self.assertFalse(login.context)
- self.client.get('/test_admin/admin/logout/')
- def testAddView(self):
- """Test add view restricts access and actually adds items."""
- add_dict = {'title': 'Døm ikke',
- 'content': '<p>great article</p>',
- 'date_0': '2008-03-18', 'date_1': '10:54:39',
- 'section': 1}
- # Change User should not have access to add articles
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.changeuser_login)
- # make sure the view removes test cookie
- self.assertEqual(self.client.session.test_cookie_worked(), False)
- response = self.client.get('/test_admin/admin/admin_views/article/add/')
- self.assertEqual(response.status_code, 403)
- # Try POST just to make sure
- post = self.client.post('/test_admin/admin/admin_views/article/add/', add_dict)
- self.assertEqual(post.status_code, 403)
- self.assertEqual(Article.objects.all().count(), 3)
- self.client.get('/test_admin/admin/logout/')
- # Add user may login and POST to add view, then redirect to admin root
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.adduser_login)
- addpage = self.client.get('/test_admin/admin/admin_views/article/add/')
- change_list_link = '› <a href="/test_admin/admin/admin_views/article/">Articles</a>'
- self.assertNotContains(addpage, change_list_link,
- msg_prefix='User restricted to add permission is given link to change list view in breadcrumbs.')
- post = self.client.post('/test_admin/admin/admin_views/article/add/', add_dict)
- self.assertRedirects(post, '/test_admin/admin/')
- self.assertEqual(Article.objects.all().count(), 4)
- self.assertEqual(len(mail.outbox), 1)
- self.assertEqual(mail.outbox[0].subject, 'Greetings from a created object')
- self.client.get('/test_admin/admin/logout/')
- # Super can add too, but is redirected to the change list view
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.super_login)
- addpage = self.client.get('/test_admin/admin/admin_views/article/add/')
- self.assertContains(addpage, change_list_link,
- msg_prefix='Unrestricted user is not given link to change list view in breadcrumbs.')
- post = self.client.post('/test_admin/admin/admin_views/article/add/', add_dict)
- self.assertRedirects(post, '/test_admin/admin/admin_views/article/')
- self.assertEqual(Article.objects.all().count(), 5)
- self.client.get('/test_admin/admin/logout/')
- # 8509 - if a normal user is already logged in, it is possible
- # to change user into the superuser without error
- login = self.client.login(username='joepublic', password='secret')
- # Check and make sure that if user expires, data still persists
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.super_login)
- # make sure the view removes test cookie
- self.assertEqual(self.client.session.test_cookie_worked(), False)
- def testChangeView(self):
- """Change view should restrict access and allow users to edit items."""
- change_dict = {'title': 'Ikke fordømt',
- 'content': '<p>edited article</p>',
- 'date_0': '2008-03-18', 'date_1': '10:54:39',
- 'section': 1}
- # add user shoud not be able to view the list of article or change any of them
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.adduser_login)
- response = self.client.get('/test_admin/admin/admin_views/article/')
- self.assertEqual(response.status_code, 403)
- response = self.client.get('/test_admin/admin/admin_views/article/1/')
- self.assertEqual(response.status_code, 403)
- post = self.client.post('/test_admin/admin/admin_views/article/1/', change_dict)
- self.assertEqual(post.status_code, 403)
- self.client.get('/test_admin/admin/logout/')
- # change user can view all items and edit them
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.changeuser_login)
- response = self.client.get('/test_admin/admin/admin_views/article/')
- self.assertEqual(response.status_code, 200)
- response = self.client.get('/test_admin/admin/admin_views/article/1/')
- self.assertEqual(response.status_code, 200)
- post = self.client.post('/test_admin/admin/admin_views/article/1/', change_dict)
- self.assertRedirects(post, '/test_admin/admin/admin_views/article/')
- self.assertEqual(Article.objects.get(pk=1).content, '<p>edited article</p>')
- # one error in form should produce singular error message, multiple errors plural
- change_dict['title'] = ''
- post = self.client.post('/test_admin/admin/admin_views/article/1/', change_dict)
- self.assertContains(post, 'Please correct the error below.',
- msg_prefix='Singular error message not found in response to post with one error')
- change_dict['content'] = ''
- post = self.client.post('/test_admin/admin/admin_views/article/1/', change_dict)
- self.assertContains(post, 'Please correct the errors below.',
- msg_prefix='Plural error message not found in response to post with multiple errors')
- self.client.get('/test_admin/admin/logout/')
- # Test redirection when using row-level change permissions. Refs #11513.
- RowLevelChangePermissionModel.objects.create(id=1, name="odd id")
- RowLevelChangePermissionModel.objects.create(id=2, name="even id")
- for login_dict in [self.super_login, self.changeuser_login, self.adduser_login, self.deleteuser_login]:
- self.client.post('/test_admin/admin/', login_dict)
- response = self.client.get('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/1/')
- self.assertEqual(response.status_code, 403)
- response = self.client.post('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/1/', {'name': 'changed'})
- self.assertEqual(RowLevelChangePermissionModel.objects.get(id=1).name, 'odd id')
- self.assertEqual(response.status_code, 403)
- response = self.client.get('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/2/')
- self.assertEqual(response.status_code, 200)
- response = self.client.post('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/2/', {'name': 'changed'})
- self.assertEqual(RowLevelChangePermissionModel.objects.get(id=2).name, 'changed')
- self.assertRedirects(response, '/test_admin/admin/')
- self.client.get('/test_admin/admin/logout/')
- for login_dict in [self.joepublic_login, self.no_username_login]:
- self.client.post('/test_admin/admin/', login_dict)
- response = self.client.get('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/1/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'login-form')
- response = self.client.post('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/1/', {'name': 'changed'})
- self.assertEqual(RowLevelChangePermissionModel.objects.get(id=1).name, 'odd id')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'login-form')
- response = self.client.get('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/2/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'login-form')
- response = self.client.post('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/2/', {'name': 'changed again'})
- self.assertEqual(RowLevelChangePermissionModel.objects.get(id=2).name, 'changed')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'login-form')
- self.client.get('/test_admin/admin/logout/')
- def testHistoryView(self):
- """History view should restrict access."""
- # add user shoud not be able to view the list of article or change any of them
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.adduser_login)
- response = self.client.get('/test_admin/admin/admin_views/article/1/history/')
- self.assertEqual(response.status_code, 403)
- self.client.get('/test_admin/admin/logout/')
- # change user can view all items and edit them
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.changeuser_login)
- response = self.client.get('/test_admin/admin/admin_views/article/1/history/')
- self.assertEqual(response.status_code, 200)
- # Test redirection when using row-level change permissions. Refs #11513.
- RowLevelChangePermissionModel.objects.create(id=1, name="odd id")
- RowLevelChangePermissionModel.objects.create(id=2, name="even id")
- for login_dict in [self.super_login, self.changeuser_login, self.adduser_login, self.deleteuser_login]:
- self.client.post('/test_admin/admin/', login_dict)
- response = self.client.get('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/1/history/')
- self.assertEqual(response.status_code, 403)
- response = self.client.get('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/2/history/')
- self.assertEqual(response.status_code, 200)
- self.client.get('/test_admin/admin/logout/')
- for login_dict in [self.joepublic_login, self.no_username_login]:
- self.client.post('/test_admin/admin/', login_dict)
- response = self.client.get('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/1/history/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'login-form')
- response = self.client.get('/test_admin/admin/admin_views/rowlevelchangepermissionmodel/2/history/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'login-form')
- self.client.get('/test_admin/admin/logout/')
- def testConditionallyShowAddSectionLink(self):
- """
- The foreign key widget should only show the "add related" button if the
- user has permission to add that related item.
- """
- # Set up and log in user.
- url = '/test_admin/admin/admin_views/article/add/'
- add_link_text = ' class="add-another"'
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.adduser_login)
- # The add user can't add sections yet, so they shouldn't see the "add
- # section" link.
- response = self.client.get(url)
- self.assertNotContains(response, add_link_text)
- # Allow the add user to add sections too. Now they can see the "add
- # section" link.
- add_user = User.objects.get(username='adduser')
- perm = get_perm(Section, get_permission_codename('add', Section._meta))
- add_user.user_permissions.add(perm)
- response = self.client.get(url)
- self.assertContains(response, add_link_text)
- def testCustomModelAdminTemplates(self):
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.super_login)
- # Test custom change list template with custom extra context
- response = self.client.get('/test_admin/admin/admin_views/customarticle/')
- self.assertContains(response, "var hello = 'Hello!';")
- self.assertTemplateUsed(response, 'custom_admin/change_list.html')
- # Test custom add form template
- response = self.client.get('/test_admin/admin/admin_views/customarticle/add/')
- self.assertTemplateUsed(response, 'custom_admin/add_form.html')
- # Add an article so we can test delete, change, and history views
- post = self.client.post('/test_admin/admin/admin_views/customarticle/add/', {
- 'content': '<p>great article</p>',
- 'date_0': '2008-03-18',
- 'date_1': '10:54:39'
- })
- self.assertRedirects(post, '/test_admin/admin/admin_views/customarticle/')
- self.assertEqual(CustomArticle.objects.all().count(), 1)
- article_pk = CustomArticle.objects.all()[0].pk
- # Test custom delete, change, and object history templates
- # Test custom change form template
- response = self.client.get('/test_admin/admin/admin_views/customarticle/%d/' % article_pk)
- self.assertTemplateUsed(response, 'custom_admin/change_form.html')
- response = self.client.get('/test_admin/admin/admin_views/customarticle/%d/delete/' % article_pk)
- self.assertTemplateUsed(response, 'custom_admin/delete_confirmation.html')
- response = self.client.post('/test_admin/admin/admin_views/customarticle/', data={
- 'index': 0,
- 'action': ['delete_selected'],
- '_selected_action': ['1'],
- })
- self.assertTemplateUsed(response, 'custom_admin/delete_selected_confirmation.html')
- response = self.client.get('/test_admin/admin/admin_views/customarticle/%d/history/' % article_pk)
- self.assertTemplateUsed(response, 'custom_admin/object_history.html')
- self.client.get('/test_admin/admin/logout/')
- def testDeleteView(self):
- """Delete view should restrict access and actually delete items."""
- delete_dict = {'post': 'yes'}
- # add user shoud not be able to delete articles
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.adduser_login)
- response = self.client.get('/test_admin/admin/admin_views/article/1/delete/')
- self.assertEqual(response.status_code, 403)
- post = self.client.post('/test_admin/admin/admin_views/article/1/delete/', delete_dict)
- self.assertEqual(post.status_code, 403)
- self.assertEqual(Article.objects.all().count(), 3)
- self.client.get('/test_admin/admin/logout/')
- # Delete user can delete
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.deleteuser_login)
- response = self.client.get('/test_admin/admin/admin_views/section/1/delete/')
- # test response contains link to related Article
- self.assertContains(response, "admin_views/article/1/")
- response = self.client.get('/test_admin/admin/admin_views/article/1/delete/')
- self.assertEqual(response.status_code, 200)
- post = self.client.post('/test_admin/admin/admin_views/article/1/delete/', delete_dict)
- self.assertRedirects(post, '/test_admin/admin/')
- self.assertEqual(Article.objects.all().count(), 2)
- self.assertEqual(len(mail.outbox), 1)
- self.assertEqual(mail.outbox[0].subject, 'Greetings from a deleted object')
- article_ct = ContentType.objects.get_for_model(Article)
- logged = LogEntry.objects.get(content_type=article_ct, action_flag=DELETION)
- self.assertEqual(logged.object_id, '1')
- self.client.get('/test_admin/admin/logout/')
- def testDisabledPermissionsWhenLoggedIn(self):
- self.client.login(username='super', password='secret')
- superuser = User.objects.get(username='super')
- superuser.is_active = False
- superuser.save()
- response = self.client.get('/test_admin/admin/')
- self.assertContains(response, 'id="login-form"')
- self.assertNotContains(response, 'Log out')
- response = self.client.get('/test_admin/admin/secure-view/')
- self.assertContains(response, 'id="login-form"')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminViewsNoUrlTest(TestCase):
- """Regression test for #17333"""
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- opts = Report._meta
- # User who can change Reports
- change_user = User.objects.get(username='changeuser')
- change_user.user_permissions.add(get_perm(Report,
- get_permission_codename('change', opts)))
- # login POST dict
- self.changeuser_login = {
- REDIRECT_FIELD_NAME: '/test_admin/admin/',
- LOGIN_FORM_KEY: 1,
- 'username': 'changeuser',
- 'password': 'secret',
- }
- def test_no_standard_modeladmin_urls(self):
- """Admin index views don't break when user's ModelAdmin removes standard urls"""
- self.client.get('/test_admin/admin/')
- self.client.post('/test_admin/admin/', self.changeuser_login)
- r = self.client.get('/test_admin/admin/')
- # we shouldn' get an 500 error caused by a NoReverseMatch
- self.assertEqual(r.status_code, 200)
- self.client.get('/test_admin/admin/logout/')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminViewDeletedObjectsTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml', 'deleted-objects.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_nesting(self):
- """
- Objects should be nested to display the relationships that
- cause them to be scheduled for deletion.
- """
- pattern = re.compile(br"""<li>Plot: <a href=".+/admin_views/plot/1/">World Domination</a>\s*<ul>\s*<li>Plot details: <a href=".+/admin_views/plotdetails/1/">almost finished</a>""")
- response = self.client.get('/test_admin/admin/admin_views/villain/%s/delete/' % quote(1))
- six.assertRegex(self, response.content, pattern)
- def test_cyclic(self):
- """
- Cyclic relationships should still cause each object to only be
- listed once.
- """
- one = """<li>Cyclic one: <a href="/test_admin/admin/admin_views/cyclicone/1/">I am recursive</a>"""
- two = """<li>Cyclic two: <a href="/test_admin/admin/admin_views/cyclictwo/1/">I am recursive too</a>"""
- response = self.client.get('/test_admin/admin/admin_views/cyclicone/%s/delete/' % quote(1))
- self.assertContains(response, one, 1)
- self.assertContains(response, two, 1)
- def test_perms_needed(self):
- self.client.logout()
- delete_user = User.objects.get(username='deleteuser')
- delete_user.user_permissions.add(get_perm(Plot,
- get_permission_codename('delete', Plot._meta)))
- self.assertTrue(self.client.login(username='deleteuser',
- password='secret'))
- response = self.client.get('/test_admin/admin/admin_views/plot/%s/delete/' % quote(1))
- self.assertContains(response, "your account doesn't have permission to delete the following types of objects")
- self.assertContains(response, "<li>plot details</li>")
- def test_protected(self):
- q = Question.objects.create(question="Why?")
- a1 = Answer.objects.create(question=q, answer="Because.")
- a2 = Answer.objects.create(question=q, answer="Yes.")
- response = self.client.get("/test_admin/admin/admin_views/question/%s/delete/" % quote(q.pk))
- self.assertContains(response, "would require deleting the following protected related objects")
- self.assertContains(response, '<li>Answer: <a href="/test_admin/admin/admin_views/answer/%s/">Because.</a></li>' % a1.pk)
- self.assertContains(response, '<li>Answer: <a href="/test_admin/admin/admin_views/answer/%s/">Yes.</a></li>' % a2.pk)
- def test_not_registered(self):
- should_contain = """<li>Secret hideout: underground bunker"""
- response = self.client.get('/test_admin/admin/admin_views/villain/%s/delete/' % quote(1))
- self.assertContains(response, should_contain, 1)
- def test_multiple_fkeys_to_same_model(self):
- """
- If a deleted object has two relationships from another model,
- both of those should be followed in looking for related
- objects to delete.
- """
- should_contain = """<li>Plot: <a href="/test_admin/admin/admin_views/plot/1/">World Domination</a>"""
- response = self.client.get('/test_admin/admin/admin_views/villain/%s/delete/' % quote(1))
- self.assertContains(response, should_contain)
- response = self.client.get('/test_admin/admin/admin_views/villain/%s/delete/' % quote(2))
- self.assertContains(response, should_contain)
- def test_multiple_fkeys_to_same_instance(self):
- """
- If a deleted object has two relationships pointing to it from
- another object, the other object should still only be listed
- once.
- """
- should_contain = """<li>Plot: <a href="/test_admin/admin/admin_views/plot/2/">World Peace</a></li>"""
- response = self.client.get('/test_admin/admin/admin_views/villain/%s/delete/' % quote(2))
- self.assertContains(response, should_contain, 1)
- def test_inheritance(self):
- """
- In the case of an inherited model, if either the child or
- parent-model instance is deleted, both instances are listed
- for deletion, as well as any relationships they have.
- """
- should_contain = [
- """<li>Villain: <a href="/test_admin/admin/admin_views/villain/3/">Bob</a>""",
- """<li>Super villain: <a href="/test_admin/admin/admin_views/supervillain/3/">Bob</a>""",
- """<li>Secret hideout: floating castle""",
- """<li>Super secret hideout: super floating castle!"""
- ]
- response = self.client.get('/test_admin/admin/admin_views/villain/%s/delete/' % quote(3))
- for should in should_contain:
- self.assertContains(response, should, 1)
- response = self.client.get('/test_admin/admin/admin_views/supervillain/%s/delete/' % quote(3))
- for should in should_contain:
- self.assertContains(response, should, 1)
- def test_generic_relations(self):
- """
- If a deleted object has GenericForeignKeys pointing to it,
- those objects should be listed for deletion.
- """
- plot = Plot.objects.get(pk=3)
- tag = FunkyTag.objects.create(content_object=plot, name='hott')
- should_contain = """<li>Funky tag: hott"""
- response = self.client.get('/test_admin/admin/admin_views/plot/%s/delete/' % quote(3))
- self.assertContains(response, should_contain)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminViewStringPrimaryKeyTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml', 'string-primary-key.xml']
- def __init__(self, *args):
- super(AdminViewStringPrimaryKeyTest, self).__init__(*args)
- self.pk = """abcdefghijklmnopqrstuvwxyz ABCDEFGHIJKLMNOPQRSTUVWXYZ 1234567890 -_.!~*'() ;/?:@&=+$, <>#%" {}|\^[]`"""
- def setUp(self):
- self.client.login(username='super', password='secret')
- content_type_pk = ContentType.objects.get_for_model(ModelWithStringPrimaryKey).pk
- LogEntry.objects.log_action(100, content_type_pk, self.pk, self.pk, 2, change_message='Changed something')
- def tearDown(self):
- self.client.logout()
- def test_get_history_view(self):
- """
- Retrieving the history for an object using urlencoded form of primary
- key should work.
- Refs #12349, #18550.
- """
- response = self.client.get('/test_admin/admin/admin_views/modelwithstringprimarykey/%s/history/' % quote(self.pk))
- self.assertContains(response, escape(self.pk))
- self.assertContains(response, 'Changed something')
- self.assertEqual(response.status_code, 200)
- def test_get_change_view(self):
- "Retrieving the object using urlencoded form of primary key should work"
- response = self.client.get('/test_admin/admin/admin_views/modelwithstringprimarykey/%s/' % quote(self.pk))
- self.assertContains(response, escape(self.pk))
- self.assertEqual(response.status_code, 200)
- def test_changelist_to_changeform_link(self):
- "Link to the changeform of the object in changelist should use reverse() and be quoted -- #18072"
- prefix = '/test_admin/admin/admin_views/modelwithstringprimarykey/'
- response = self.client.get(prefix)
- # this URL now comes through reverse(), thus url quoting and iri_to_uri encoding
- pk_final_url = escape(iri_to_uri(urlquote(quote(self.pk))))
- should_contain = """<th class="field-__str__"><a href="%s%s/">%s</a></th>""" % (prefix, pk_final_url, escape(self.pk))
- self.assertContains(response, should_contain)
- def test_recentactions_link(self):
- "The link from the recent actions list referring to the changeform of the object should be quoted"
- response = self.client.get('/test_admin/admin/')
- link = reverse('admin:admin_views_modelwithstringprimarykey_change', args=(quote(self.pk),))
- should_contain = """<a href="%s">%s</a>""" % (link, escape(self.pk))
- self.assertContains(response, should_contain)
- def test_recentactions_without_content_type(self):
- "If a LogEntry is missing content_type it will not display it in span tag under the hyperlink."
- response = self.client.get('/test_admin/admin/')
- link = reverse('admin:admin_views_modelwithstringprimarykey_change', args=(quote(self.pk),))
- should_contain = """<a href="%s">%s</a>""" % (link, escape(self.pk))
- self.assertContains(response, should_contain)
- should_contain = "Model with string primary key" # capitalized in Recent Actions
- self.assertContains(response, should_contain)
- logentry = LogEntry.objects.get(content_type__name__iexact=should_contain)
- # http://code.djangoproject.com/ticket/10275
- # if the log entry doesn't have a content type it should still be
- # possible to view the Recent Actions part
- logentry.content_type = None
- logentry.save()
- counted_presence_before = response.content.count(force_bytes(should_contain))
- response = self.client.get('/test_admin/admin/')
- counted_presence_after = response.content.count(force_bytes(should_contain))
- self.assertEqual(counted_presence_before - 1,
- counted_presence_after)
- def test_logentry_get_admin_url(self):
- "LogEntry.get_admin_url returns a URL to edit the entry's object or None for non-existent (possibly deleted) models"
- log_entry_name = "Model with string primary key" # capitalized in Recent Actions
- logentry = LogEntry.objects.get(content_type__name__iexact=log_entry_name)
- model = "modelwithstringprimarykey"
- desired_admin_url = "/test_admin/admin/admin_views/%s/%s/" % (model, escape(iri_to_uri(urlquote(quote(self.pk)))))
- self.assertEqual(logentry.get_admin_url(), desired_admin_url)
- logentry.content_type.model = "non-existent"
- self.assertEqual(logentry.get_admin_url(), None)
- def test_deleteconfirmation_link(self):
- "The link from the delete confirmation page referring back to the changeform of the object should be quoted"
- response = self.client.get('/test_admin/admin/admin_views/modelwithstringprimarykey/%s/delete/' % quote(self.pk))
- # this URL now comes through reverse(), thus url quoting and iri_to_uri encoding
- should_contain = """/%s/">%s</a>""" % (escape(iri_to_uri(urlquote(quote(self.pk)))), escape(self.pk))
- self.assertContains(response, should_contain)
- def test_url_conflicts_with_add(self):
- "A model with a primary key that ends with add should be visible"
- add_model = ModelWithStringPrimaryKey(pk="i have something to add")
- add_model.save()
- response = self.client.get('/test_admin/admin/admin_views/modelwithstringprimarykey/%s/' % quote(add_model.pk))
- should_contain = """<h1>Change model with string primary key</h1>"""
- self.assertContains(response, should_contain)
- def test_url_conflicts_with_delete(self):
- "A model with a primary key that ends with delete should be visible"
- delete_model = ModelWithStringPrimaryKey(pk="delete")
- delete_model.save()
- response = self.client.get('/test_admin/admin/admin_views/modelwithstringprimarykey/%s/' % quote(delete_model.pk))
- should_contain = """<h1>Change model with string primary key</h1>"""
- self.assertContains(response, should_contain)
- def test_url_conflicts_with_history(self):
- "A model with a primary key that ends with history should be visible"
- history_model = ModelWithStringPrimaryKey(pk="history")
- history_model.save()
- response = self.client.get('/test_admin/admin/admin_views/modelwithstringprimarykey/%s/' % quote(history_model.pk))
- should_contain = """<h1>Change model with string primary key</h1>"""
- self.assertContains(response, should_contain)
- def test_shortcut_view_with_escaping(self):
- "'View on site should' work properly with char fields"
- model = ModelWithStringPrimaryKey(pk='abc_123')
- model.save()
- response = self.client.get('/test_admin/admin/admin_views/modelwithstringprimarykey/%s/' % quote(model.pk))
- should_contain = '/%s/" class="viewsitelink">' % model.pk
- self.assertContains(response, should_contain)
- def test_change_view_history_link(self):
- """Object history button link should work and contain the pk value quoted."""
- url = reverse('admin:%s_modelwithstringprimarykey_change' %
- ModelWithStringPrimaryKey._meta.app_label,
- args=(quote(self.pk),))
- response = self.client.get(url)
- self.assertEqual(response.status_code, 200)
- expected_link = reverse('admin:%s_modelwithstringprimarykey_history' %
- ModelWithStringPrimaryKey._meta.app_label,
- args=(quote(self.pk),))
- self.assertContains(response, '<a href="%s" class="historylink"' % expected_link)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class SecureViewTests(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- # login POST dicts
- self.super_login = {
- LOGIN_FORM_KEY: 1,
- REDIRECT_FIELD_NAME: '/test_admin/admin/secure-view/',
- 'username': 'super',
- 'password': 'secret',
- }
- self.super_email_login = {
- LOGIN_FORM_KEY: 1,
- REDIRECT_FIELD_NAME: '/test_admin/admin/secure-view/',
- 'username': 'super@example.com',
- 'password': 'secret',
- }
- self.super_email_bad_login = {
- LOGIN_FORM_KEY: 1,
- REDIRECT_FIELD_NAME: '/test_admin/admin/secure-view/',
- 'username': 'super@example.com',
- 'password': 'notsecret',
- }
- self.adduser_login = {
- LOGIN_FORM_KEY: 1,
- REDIRECT_FIELD_NAME: '/test_admin/admin/secure-view/',
- 'username': 'adduser',
- 'password': 'secret',
- }
- self.changeuser_login = {
- LOGIN_FORM_KEY: 1,
- REDIRECT_FIELD_NAME: '/test_admin/admin/secure-view/',
- 'username': 'changeuser',
- 'password': 'secret',
- }
- self.deleteuser_login = {
- LOGIN_FORM_KEY: 1,
- REDIRECT_FIELD_NAME: '/test_admin/admin/secure-view/',
- 'username': 'deleteuser',
- 'password': 'secret',
- }
- self.joepublic_login = {
- LOGIN_FORM_KEY: 1,
- REDIRECT_FIELD_NAME: '/test_admin/admin/secure-view/',
- 'username': 'joepublic',
- 'password': 'secret',
- }
- def tearDown(self):
- self.client.logout()
- def test_secure_view_shows_login_if_not_logged_in(self):
- "Ensure that we see the login form"
- response = self.client.get('/test_admin/admin/secure-view/')
- self.assertTemplateUsed(response, 'admin/login.html')
- def test_secure_view_login_successfully_redirects_to_original_url(self):
- response = self.client.get('/test_admin/admin/secure-view/')
- self.assertEqual(response.status_code, 200)
- query_string = 'the-answer=42'
- redirect_url = '/test_admin/admin/secure-view/?%s' % query_string
- new_next = {REDIRECT_FIELD_NAME: redirect_url}
- login = self.client.post('/test_admin/admin/secure-view/', dict(self.super_login, **new_next), QUERY_STRING=query_string)
- self.assertRedirects(login, redirect_url)
- def test_staff_member_required_decorator_works_as_per_admin_login(self):
- """
- Make sure only staff members can log in.
- Successful posts to the login page will redirect to the orignal url.
- Unsuccessfull attempts will continue to render the login page with
- a 200 status code.
- """
- # Super User
- response = self.client.get('/test_admin/admin/secure-view/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/secure-view/', self.super_login)
- self.assertRedirects(login, '/test_admin/admin/secure-view/')
- self.assertFalse(login.context)
- self.client.get('/test_admin/admin/logout/')
- # make sure the view removes test cookie
- self.assertEqual(self.client.session.test_cookie_worked(), False)
- # Test if user enters email address
- response = self.client.get('/test_admin/admin/secure-view/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/secure-view/', self.super_email_login)
- self.assertContains(login, ERROR_MESSAGE)
- # only correct passwords get a username hint
- login = self.client.post('/test_admin/admin/secure-view/', self.super_email_bad_login)
- self.assertContains(login, ERROR_MESSAGE)
- new_user = User(username='jondoe', password='secret', email='super@example.com')
- new_user.save()
- # check to ensure if there are multiple email addresses a user doesn't get a 500
- login = self.client.post('/test_admin/admin/secure-view/', self.super_email_login)
- self.assertContains(login, ERROR_MESSAGE)
- # Add User
- response = self.client.get('/test_admin/admin/secure-view/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/secure-view/', self.adduser_login)
- self.assertRedirects(login, '/test_admin/admin/secure-view/')
- self.assertFalse(login.context)
- self.client.get('/test_admin/admin/logout/')
- # Change User
- response = self.client.get('/test_admin/admin/secure-view/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/secure-view/', self.changeuser_login)
- self.assertRedirects(login, '/test_admin/admin/secure-view/')
- self.assertFalse(login.context)
- self.client.get('/test_admin/admin/logout/')
- # Delete User
- response = self.client.get('/test_admin/admin/secure-view/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/secure-view/', self.deleteuser_login)
- self.assertRedirects(login, '/test_admin/admin/secure-view/')
- self.assertFalse(login.context)
- self.client.get('/test_admin/admin/logout/')
- # Regular User should not be able to login.
- response = self.client.get('/test_admin/admin/secure-view/')
- self.assertEqual(response.status_code, 200)
- login = self.client.post('/test_admin/admin/secure-view/', self.joepublic_login)
- self.assertEqual(login.status_code, 200)
- # Login.context is a list of context dicts we just need to check the first one.
- self.assertContains(login, ERROR_MESSAGE)
- # 8509 - if a normal user is already logged in, it is possible
- # to change user into the superuser without error
- login = self.client.login(username='joepublic', password='secret')
- # Check and make sure that if user expires, data still persists
- self.client.get('/test_admin/admin/secure-view/')
- self.client.post('/test_admin/admin/secure-view/', self.super_login)
- # make sure the view removes test cookie
- self.assertEqual(self.client.session.test_cookie_worked(), False)
- def test_shortcut_view_only_available_to_staff(self):
- """
- Only admin users should be able to use the admin shortcut view.
- """
- user_ctype = ContentType.objects.get_for_model(User)
- user = User.objects.get(username='super')
- shortcut_url = "/test_admin/admin/r/%s/%s/" % (user_ctype.pk, user.pk)
- # Not logged in: we should see the login page.
- response = self.client.get(shortcut_url, follow=False)
- self.assertTemplateUsed(response, 'admin/login.html')
- # Logged in? Redirect.
- self.client.login(username='super', password='secret')
- response = self.client.get(shortcut_url, follow=False)
- # Can't use self.assertRedirects() because User.get_absolute_url() is silly.
- self.assertEqual(response.status_code, 302)
- self.assertEqual(response.url, 'http://example.com/users/super/')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminViewUnicodeTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-unicode.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def testUnicodeEdit(self):
- """
- A test to ensure that POST on edit_view handles non-ascii characters.
- """
- post_data = {
- "name": "Test lærdommer",
- # inline data
- "chapter_set-TOTAL_FORMS": "6",
- "chapter_set-INITIAL_FORMS": "3",
- "chapter_set-MAX_NUM_FORMS": "0",
- "chapter_set-0-id": "1",
- "chapter_set-0-title": "Norske bostaver æøå skaper problemer",
- "chapter_set-0-content": "<p>Svært frustrerende med UnicodeDecodeError</p>",
- "chapter_set-1-id": "2",
- "chapter_set-1-title": "Kjærlighet.",
- "chapter_set-1-content": "<p>La kjærligheten til de lidende seire.</p>",
- "chapter_set-2-id": "3",
- "chapter_set-2-title": "Need a title.",
- "chapter_set-2-content": "<p>Newest content</p>",
- "chapter_set-3-id": "",
- "chapter_set-3-title": "",
- "chapter_set-3-content": "",
- "chapter_set-4-id": "",
- "chapter_set-4-title": "",
- "chapter_set-4-content": "",
- "chapter_set-5-id": "",
- "chapter_set-5-title": "",
- "chapter_set-5-content": "",
- }
- response = self.client.post('/test_admin/admin/admin_views/book/1/', post_data)
- self.assertEqual(response.status_code, 302) # redirect somewhere
- def testUnicodeDelete(self):
- """
- Ensure that the delete_view handles non-ascii characters
- """
- delete_dict = {'post': 'yes'}
- response = self.client.get('/test_admin/admin/admin_views/book/1/delete/')
- self.assertEqual(response.status_code, 200)
- response = self.client.post('/test_admin/admin/admin_views/book/1/delete/', delete_dict)
- self.assertRedirects(response, '/test_admin/admin/admin_views/book/')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminViewListEditable(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml', 'admin-views-person.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_inheritance(self):
- Podcast.objects.create(name="This Week in Django",
- release_date=datetime.date.today())
- response = self.client.get('/test_admin/admin/admin_views/podcast/')
- self.assertEqual(response.status_code, 200)
- def test_inheritance_2(self):
- Vodcast.objects.create(name="This Week in Django", released=True)
- response = self.client.get('/test_admin/admin/admin_views/vodcast/')
- self.assertEqual(response.status_code, 200)
- def test_custom_pk(self):
- Language.objects.create(iso='en', name='English', english_name='English')
- response = self.client.get('/test_admin/admin/admin_views/language/')
- self.assertEqual(response.status_code, 200)
- def test_changelist_input_html(self):
- response = self.client.get('/test_admin/admin/admin_views/person/')
- # 2 inputs per object(the field and the hidden id field) = 6
- # 3 management hidden fields = 3
- # 4 action inputs (3 regular checkboxes, 1 checkbox to select all)
- # main form submit button = 1
- # search field and search submit button = 2
- # CSRF field = 1
- # field to track 'select all' across paginated views = 1
- # 6 + 3 + 4 + 1 + 2 + 1 + 1 = 18 inputs
- self.assertContains(response, "<input", count=18)
- # 1 select per object = 3 selects
- self.assertContains(response, "<select", count=4)
- def test_post_messages(self):
- # Ticket 12707: Saving inline editable should not show admin
- # action warnings
- data = {
- "form-TOTAL_FORMS": "3",
- "form-INITIAL_FORMS": "3",
- "form-MAX_NUM_FORMS": "0",
- "form-0-gender": "1",
- "form-0-id": "1",
- "form-1-gender": "2",
- "form-1-id": "2",
- "form-2-alive": "checked",
- "form-2-gender": "1",
- "form-2-id": "3",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/person/',
- data, follow=True)
- self.assertEqual(len(response.context['messages']), 1)
- def test_post_submission(self):
- data = {
- "form-TOTAL_FORMS": "3",
- "form-INITIAL_FORMS": "3",
- "form-MAX_NUM_FORMS": "0",
- "form-0-gender": "1",
- "form-0-id": "1",
- "form-1-gender": "2",
- "form-1-id": "2",
- "form-2-alive": "checked",
- "form-2-gender": "1",
- "form-2-id": "3",
- "_save": "Save",
- }
- self.client.post('/test_admin/admin/admin_views/person/', data)
- self.assertEqual(Person.objects.get(name="John Mauchly").alive, False)
- self.assertEqual(Person.objects.get(name="Grace Hopper").gender, 2)
- # test a filtered page
- data = {
- "form-TOTAL_FORMS": "2",
- "form-INITIAL_FORMS": "2",
- "form-MAX_NUM_FORMS": "0",
- "form-0-id": "1",
- "form-0-gender": "1",
- "form-0-alive": "checked",
- "form-1-id": "3",
- "form-1-gender": "1",
- "form-1-alive": "checked",
- "_save": "Save",
- }
- self.client.post('/test_admin/admin/admin_views/person/?gender__exact=1', data)
- self.assertEqual(Person.objects.get(name="John Mauchly").alive, True)
- # test a searched page
- data = {
- "form-TOTAL_FORMS": "1",
- "form-INITIAL_FORMS": "1",
- "form-MAX_NUM_FORMS": "0",
- "form-0-id": "1",
- "form-0-gender": "1",
- "_save": "Save",
- }
- self.client.post('/test_admin/admin/admin_views/person/?q=john', data)
- self.assertEqual(Person.objects.get(name="John Mauchly").alive, False)
- def test_non_field_errors(self):
- ''' Ensure that non field errors are displayed for each of the
- forms in the changelist's formset. Refs #13126.
- '''
- fd1 = FoodDelivery.objects.create(reference='123', driver='bill', restaurant='thai')
- fd2 = FoodDelivery.objects.create(reference='456', driver='bill', restaurant='india')
- fd3 = FoodDelivery.objects.create(reference='789', driver='bill', restaurant='pizza')
- data = {
- "form-TOTAL_FORMS": "3",
- "form-INITIAL_FORMS": "3",
- "form-MAX_NUM_FORMS": "0",
- "form-0-id": str(fd1.id),
- "form-0-reference": "123",
- "form-0-driver": "bill",
- "form-0-restaurant": "thai",
- # Same data as above: Forbidden because of unique_together!
- "form-1-id": str(fd2.id),
- "form-1-reference": "456",
- "form-1-driver": "bill",
- "form-1-restaurant": "thai",
- "form-2-id": str(fd3.id),
- "form-2-reference": "789",
- "form-2-driver": "bill",
- "form-2-restaurant": "pizza",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/fooddelivery/', data)
- self.assertContains(response, '<tr><td colspan="4"><ul class="errorlist"><li>Food delivery with this Driver and Restaurant already exists.</li></ul></td></tr>', 1, html=True)
- data = {
- "form-TOTAL_FORMS": "3",
- "form-INITIAL_FORMS": "3",
- "form-MAX_NUM_FORMS": "0",
- "form-0-id": str(fd1.id),
- "form-0-reference": "123",
- "form-0-driver": "bill",
- "form-0-restaurant": "thai",
- # Same data as above: Forbidden because of unique_together!
- "form-1-id": str(fd2.id),
- "form-1-reference": "456",
- "form-1-driver": "bill",
- "form-1-restaurant": "thai",
- # Same data also.
- "form-2-id": str(fd3.id),
- "form-2-reference": "789",
- "form-2-driver": "bill",
- "form-2-restaurant": "thai",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/fooddelivery/', data)
- self.assertContains(response, '<tr><td colspan="4"><ul class="errorlist"><li>Food delivery with this Driver and Restaurant already exists.</li></ul></td></tr>', 2, html=True)
- def test_non_form_errors(self):
- # test if non-form errors are handled; ticket #12716
- data = {
- "form-TOTAL_FORMS": "1",
- "form-INITIAL_FORMS": "1",
- "form-MAX_NUM_FORMS": "0",
- "form-0-id": "2",
- "form-0-alive": "1",
- "form-0-gender": "2",
- # Ensure that the form processing understands this as a list_editable "Save"
- # and not an action "Go".
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/person/', data)
- self.assertContains(response, "Grace is not a Zombie")
- def test_non_form_errors_is_errorlist(self):
- # test if non-form errors are correctly handled; ticket #12878
- data = {
- "form-TOTAL_FORMS": "1",
- "form-INITIAL_FORMS": "1",
- "form-MAX_NUM_FORMS": "0",
- "form-0-id": "2",
- "form-0-alive": "1",
- "form-0-gender": "2",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/person/', data)
- non_form_errors = response.context['cl'].formset.non_form_errors()
- self.assertIsInstance(non_form_errors, ErrorList)
- self.assertEqual(str(non_form_errors), str(ErrorList(["Grace is not a Zombie"])))
- def test_list_editable_ordering(self):
- collector = Collector.objects.create(id=1, name="Frederick Clegg")
- Category.objects.create(id=1, order=1, collector=collector)
- Category.objects.create(id=2, order=2, collector=collector)
- Category.objects.create(id=3, order=0, collector=collector)
- Category.objects.create(id=4, order=0, collector=collector)
- # NB: The order values must be changed so that the items are reordered.
- data = {
- "form-TOTAL_FORMS": "4",
- "form-INITIAL_FORMS": "4",
- "form-MAX_NUM_FORMS": "0",
- "form-0-order": "14",
- "form-0-id": "1",
- "form-0-collector": "1",
- "form-1-order": "13",
- "form-1-id": "2",
- "form-1-collector": "1",
- "form-2-order": "1",
- "form-2-id": "3",
- "form-2-collector": "1",
- "form-3-order": "0",
- "form-3-id": "4",
- "form-3-collector": "1",
- # Ensure that the form processing understands this as a list_editable "Save"
- # and not an action "Go".
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/category/', data)
- # Successful post will redirect
- self.assertEqual(response.status_code, 302)
- # Check that the order values have been applied to the right objects
- self.assertEqual(Category.objects.get(id=1).order, 14)
- self.assertEqual(Category.objects.get(id=2).order, 13)
- self.assertEqual(Category.objects.get(id=3).order, 1)
- self.assertEqual(Category.objects.get(id=4).order, 0)
- def test_list_editable_pagination(self):
- """
- Ensure that pagination works for list_editable items.
- Refs #16819.
- """
- UnorderedObject.objects.create(id=1, name='Unordered object #1')
- UnorderedObject.objects.create(id=2, name='Unordered object #2')
- UnorderedObject.objects.create(id=3, name='Unordered object #3')
- response = self.client.get('/test_admin/admin/admin_views/unorderedobject/')
- self.assertContains(response, 'Unordered object #3')
- self.assertContains(response, 'Unordered object #2')
- self.assertNotContains(response, 'Unordered object #1')
- response = self.client.get('/test_admin/admin/admin_views/unorderedobject/?p=1')
- self.assertNotContains(response, 'Unordered object #3')
- self.assertNotContains(response, 'Unordered object #2')
- self.assertContains(response, 'Unordered object #1')
- def test_list_editable_action_submit(self):
- # List editable changes should not be executed if the action "Go" button is
- # used to submit the form.
- data = {
- "form-TOTAL_FORMS": "3",
- "form-INITIAL_FORMS": "3",
- "form-MAX_NUM_FORMS": "0",
- "form-0-gender": "1",
- "form-0-id": "1",
- "form-1-gender": "2",
- "form-1-id": "2",
- "form-2-alive": "checked",
- "form-2-gender": "1",
- "form-2-id": "3",
- "index": "0",
- "_selected_action": ['3'],
- "action": ['', 'delete_selected'],
- }
- self.client.post('/test_admin/admin/admin_views/person/', data)
- self.assertEqual(Person.objects.get(name="John Mauchly").alive, True)
- self.assertEqual(Person.objects.get(name="Grace Hopper").gender, 1)
- def test_list_editable_action_choices(self):
- # List editable changes should be executed if the "Save" button is
- # used to submit the form - any action choices should be ignored.
- data = {
- "form-TOTAL_FORMS": "3",
- "form-INITIAL_FORMS": "3",
- "form-MAX_NUM_FORMS": "0",
- "form-0-gender": "1",
- "form-0-id": "1",
- "form-1-gender": "2",
- "form-1-id": "2",
- "form-2-alive": "checked",
- "form-2-gender": "1",
- "form-2-id": "3",
- "_save": "Save",
- "_selected_action": ['1'],
- "action": ['', 'delete_selected'],
- }
- self.client.post('/test_admin/admin/admin_views/person/', data)
- self.assertEqual(Person.objects.get(name="John Mauchly").alive, False)
- self.assertEqual(Person.objects.get(name="Grace Hopper").gender, 2)
- def test_list_editable_popup(self):
- """
- Fields should not be list-editable in popups.
- """
- response = self.client.get('/test_admin/admin/admin_views/person/')
- self.assertNotEqual(response.context['cl'].list_editable, ())
- response = self.client.get('/test_admin/admin/admin_views/person/?%s' % IS_POPUP_VAR)
- self.assertEqual(response.context['cl'].list_editable, ())
- def test_pk_hidden_fields(self):
- """ Ensure that hidden pk fields aren't displayed in the table body and
- that their corresponding human-readable value is displayed instead.
- Note that the hidden pk fields are in fact be displayed but
- separately (not in the table), and only once.
- Refs #12475.
- """
- story1 = Story.objects.create(title='The adventures of Guido', content='Once upon a time in Djangoland...')
- story2 = Story.objects.create(title='Crouching Tiger, Hidden Python', content='The Python was sneaking into...')
- response = self.client.get('/test_admin/admin/admin_views/story/')
- self.assertContains(response, 'id="id_form-0-id"', 1) # Only one hidden field, in a separate place than the table.
- self.assertContains(response, 'id="id_form-1-id"', 1)
- self.assertContains(response, '<div class="hiddenfields">\n<input type="hidden" name="form-0-id" value="%d" id="id_form-0-id" /><input type="hidden" name="form-1-id" value="%d" id="id_form-1-id" />\n</div>' % (story2.id, story1.id), html=True)
- self.assertContains(response, '<td class="field-id">%d</td>' % story1.id, 1)
- self.assertContains(response, '<td class="field-id">%d</td>' % story2.id, 1)
- def test_pk_hidden_fields_with_list_display_links(self):
- """ Similarly as test_pk_hidden_fields, but when the hidden pk fields are
- referenced in list_display_links.
- Refs #12475.
- """
- story1 = OtherStory.objects.create(title='The adventures of Guido', content='Once upon a time in Djangoland...')
- story2 = OtherStory.objects.create(title='Crouching Tiger, Hidden Python', content='The Python was sneaking into...')
- link1 = reverse('admin:admin_views_otherstory_change', args=(story1.pk,))
- link2 = reverse('admin:admin_views_otherstory_change', args=(story2.pk,))
- response = self.client.get('/test_admin/admin/admin_views/otherstory/')
- self.assertContains(response, 'id="id_form-0-id"', 1) # Only one hidden field, in a separate place than the table.
- self.assertContains(response, 'id="id_form-1-id"', 1)
- self.assertContains(response, '<div class="hiddenfields">\n<input type="hidden" name="form-0-id" value="%d" id="id_form-0-id" /><input type="hidden" name="form-1-id" value="%d" id="id_form-1-id" />\n</div>' % (story2.id, story1.id), html=True)
- self.assertContains(response, '<th class="field-id"><a href="%s">%d</a></th>' % (link1, story1.id), 1)
- self.assertContains(response, '<th class="field-id"><a href="%s">%d</a></th>' % (link2, story2.id), 1)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminSearchTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users', 'multiple-child-classes',
- 'admin-views-person']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_search_on_sibling_models(self):
- "Check that a search that mentions sibling models"
- response = self.client.get('/test_admin/admin/admin_views/recommendation/?q=bar')
- # confirm the search returned 1 object
- self.assertContains(response, "\n1 recommendation\n")
- def test_with_fk_to_field(self):
- """Ensure that the to_field GET parameter is preserved when a search
- is performed. Refs #10918.
- """
- from django.contrib.admin.views.main import TO_FIELD_VAR
- response = self.client.get('/test_admin/admin/auth/user/?q=joe&%s=username' % TO_FIELD_VAR)
- self.assertContains(response, "\n1 user\n")
- self.assertContains(response, '<input type="hidden" name="t" value="username"/>', html=True)
- def test_exact_matches(self):
- response = self.client.get('/test_admin/admin/admin_views/recommendation/?q=bar')
- # confirm the search returned one object
- self.assertContains(response, "\n1 recommendation\n")
- response = self.client.get('/test_admin/admin/admin_views/recommendation/?q=ba')
- # confirm the search returned zero objects
- self.assertContains(response, "\n0 recommendations\n")
- def test_beginning_matches(self):
- response = self.client.get('/test_admin/admin/admin_views/person/?q=Gui')
- # confirm the search returned one object
- self.assertContains(response, "\n1 person\n")
- self.assertContains(response, "Guido")
- response = self.client.get('/test_admin/admin/admin_views/person/?q=uido')
- # confirm the search returned zero objects
- self.assertContains(response, "\n0 persons\n")
- self.assertNotContains(response, "Guido")
- def test_pluggable_search(self):
- p1 = PluggableSearchPerson.objects.create(name="Bob", age=10)
- p2 = PluggableSearchPerson.objects.create(name="Amy", age=20)
- response = self.client.get('/test_admin/admin/admin_views/pluggablesearchperson/?q=Bob')
- # confirm the search returned one object
- self.assertContains(response, "\n1 pluggable search person\n")
- self.assertContains(response, "Bob")
- response = self.client.get('/test_admin/admin/admin_views/pluggablesearchperson/?q=20')
- # confirm the search returned one object
- self.assertContains(response, "\n1 pluggable search person\n")
- self.assertContains(response, "Amy")
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminInheritedInlinesTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def testInline(self):
- "Ensure that inline models which inherit from a common parent are correctly handled by admin."
- foo_user = "foo username"
- bar_user = "bar username"
- name_re = re.compile(b'name="(.*?)"')
- # test the add case
- response = self.client.get('/test_admin/admin/admin_views/persona/add/')
- names = name_re.findall(response.content)
- # make sure we have no duplicate HTML names
- self.assertEqual(len(names), len(set(names)))
- # test the add case
- post_data = {
- "name": "Test Name",
- # inline data
- "accounts-TOTAL_FORMS": "1",
- "accounts-INITIAL_FORMS": "0",
- "accounts-MAX_NUM_FORMS": "0",
- "accounts-0-username": foo_user,
- "accounts-2-TOTAL_FORMS": "1",
- "accounts-2-INITIAL_FORMS": "0",
- "accounts-2-MAX_NUM_FORMS": "0",
- "accounts-2-0-username": bar_user,
- }
- response = self.client.post('/test_admin/admin/admin_views/persona/add/', post_data)
- self.assertEqual(response.status_code, 302) # redirect somewhere
- self.assertEqual(Persona.objects.count(), 1)
- self.assertEqual(FooAccount.objects.count(), 1)
- self.assertEqual(BarAccount.objects.count(), 1)
- self.assertEqual(FooAccount.objects.all()[0].username, foo_user)
- self.assertEqual(BarAccount.objects.all()[0].username, bar_user)
- self.assertEqual(Persona.objects.all()[0].accounts.count(), 2)
- persona_id = Persona.objects.all()[0].id
- foo_id = FooAccount.objects.all()[0].id
- bar_id = BarAccount.objects.all()[0].id
- # test the edit case
- response = self.client.get('/test_admin/admin/admin_views/persona/%d/' % persona_id)
- names = name_re.findall(response.content)
- # make sure we have no duplicate HTML names
- self.assertEqual(len(names), len(set(names)))
- post_data = {
- "name": "Test Name",
- "accounts-TOTAL_FORMS": "2",
- "accounts-INITIAL_FORMS": "1",
- "accounts-MAX_NUM_FORMS": "0",
- "accounts-0-username": "%s-1" % foo_user,
- "accounts-0-account_ptr": str(foo_id),
- "accounts-0-persona": str(persona_id),
- "accounts-2-TOTAL_FORMS": "2",
- "accounts-2-INITIAL_FORMS": "1",
- "accounts-2-MAX_NUM_FORMS": "0",
- "accounts-2-0-username": "%s-1" % bar_user,
- "accounts-2-0-account_ptr": str(bar_id),
- "accounts-2-0-persona": str(persona_id),
- }
- response = self.client.post('/test_admin/admin/admin_views/persona/%d/' % persona_id, post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Persona.objects.count(), 1)
- self.assertEqual(FooAccount.objects.count(), 1)
- self.assertEqual(BarAccount.objects.count(), 1)
- self.assertEqual(FooAccount.objects.all()[0].username, "%s-1" % foo_user)
- self.assertEqual(BarAccount.objects.all()[0].username, "%s-1" % bar_user)
- self.assertEqual(Persona.objects.all()[0].accounts.count(), 2)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminActionsTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml', 'admin-views-actions.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_model_admin_custom_action(self):
- "Tests a custom action defined in a ModelAdmin method"
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- 'action': 'mail_admin',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/subscriber/', action_data)
- self.assertEqual(len(mail.outbox), 1)
- self.assertEqual(mail.outbox[0].subject, 'Greetings from a ModelAdmin action')
- def test_model_admin_default_delete_action(self):
- "Tests the default delete action defined as a ModelAdmin method"
- action_data = {
- ACTION_CHECKBOX_NAME: [1, 2],
- 'action': 'delete_selected',
- 'index': 0,
- }
- delete_confirmation_data = {
- ACTION_CHECKBOX_NAME: [1, 2],
- 'action': 'delete_selected',
- 'post': 'yes',
- }
- confirmation = self.client.post('/test_admin/admin/admin_views/subscriber/', action_data)
- self.assertIsInstance(confirmation, TemplateResponse)
- self.assertContains(confirmation, "Are you sure you want to delete the selected subscribers?")
- self.assertContains(confirmation, ACTION_CHECKBOX_NAME, count=2)
- response = self.client.post('/test_admin/admin/admin_views/subscriber/', delete_confirmation_data)
- self.assertEqual(Subscriber.objects.count(), 0)
- def test_non_localized_pk(self):
- """If USE_THOUSAND_SEPARATOR is set, make sure that the ids for
- the objects selected for deletion are rendered without separators.
- Refs #14895.
- """
- self.old_USE_THOUSAND_SEPARATOR = settings.USE_THOUSAND_SEPARATOR
- self.old_USE_L10N = settings.USE_L10N
- settings.USE_THOUSAND_SEPARATOR = True
- settings.USE_L10N = True
- subscriber = Subscriber.objects.get(id=1)
- subscriber.id = 9999
- subscriber.save()
- action_data = {
- ACTION_CHECKBOX_NAME: [9999, 2],
- 'action': 'delete_selected',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/subscriber/', action_data)
- self.assertTemplateUsed(response, 'admin/delete_selected_confirmation.html')
- self.assertContains(response, 'value="9999"') # Instead of 9,999
- self.assertContains(response, 'value="2"')
- settings.USE_THOUSAND_SEPARATOR = self.old_USE_THOUSAND_SEPARATOR
- settings.USE_L10N = self.old_USE_L10N
- def test_model_admin_default_delete_action_protected(self):
- """
- Tests the default delete action defined as a ModelAdmin method in the
- case where some related objects are protected from deletion.
- """
- q1 = Question.objects.create(question="Why?")
- a1 = Answer.objects.create(question=q1, answer="Because.")
- a2 = Answer.objects.create(question=q1, answer="Yes.")
- q2 = Question.objects.create(question="Wherefore?")
- action_data = {
- ACTION_CHECKBOX_NAME: [q1.pk, q2.pk],
- 'action': 'delete_selected',
- 'index': 0,
- }
- response = self.client.post("/test_admin/admin/admin_views/question/", action_data)
- self.assertContains(response, "would require deleting the following protected related objects")
- self.assertContains(response, '<li>Answer: <a href="/test_admin/admin/admin_views/answer/%s/">Because.</a></li>' % a1.pk, html=True)
- self.assertContains(response, '<li>Answer: <a href="/test_admin/admin/admin_views/answer/%s/">Yes.</a></li>' % a2.pk, html=True)
- def test_model_admin_default_delete_action_no_change_url(self):
- """
- Default delete action shouldn't break if a user's ModelAdmin removes the url for change_view.
- Regression test for #20640
- """
- obj = UnchangeableObject.objects.create()
- action_data = {
- ACTION_CHECKBOX_NAME: obj.pk,
- "action": "delete_selected",
- "index": "0",
- }
- response = self.client.post('/test_admin/admin/admin_views/unchangeableobject/', action_data)
- # No 500 caused by NoReverseMatch
- self.assertEqual(response.status_code, 200)
- # The page shouldn't display a link to the nonexistent change page
- self.assertContains(response, "<li>Unchangeable object: UnchangeableObject object</li>", 1, html=True)
- def test_custom_function_mail_action(self):
- "Tests a custom action defined in a function"
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- 'action': 'external_mail',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/externalsubscriber/', action_data)
- self.assertEqual(len(mail.outbox), 1)
- self.assertEqual(mail.outbox[0].subject, 'Greetings from a function action')
- def test_custom_function_action_with_redirect(self):
- "Tests a custom action defined in a function"
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- 'action': 'redirect_to',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/externalsubscriber/', action_data)
- self.assertEqual(response.status_code, 302)
- def test_default_redirect(self):
- """
- Test that actions which don't return an HttpResponse are redirected to
- the same page, retaining the querystring (which may contain changelist
- information).
- """
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- 'action': 'external_mail',
- 'index': 0,
- }
- url = '/test_admin/admin/admin_views/externalsubscriber/?o=1'
- response = self.client.post(url, action_data)
- self.assertRedirects(response, url)
- def test_custom_function_action_streaming_response(self):
- """Tests a custom action that returns a StreamingHttpResponse."""
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- 'action': 'download',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/externalsubscriber/', action_data)
- content = b''.join(response.streaming_content)
- self.assertEqual(content, b'This is the content of the file')
- self.assertEqual(response.status_code, 200)
- def test_custom_function_action_no_perm_response(self):
- """Tests a custom action that returns an HttpResponse with 403 code."""
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- 'action': 'no_perm',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/externalsubscriber/', action_data)
- self.assertEqual(response.status_code, 403)
- self.assertEqual(response.content, b'No permission to perform this action')
- def test_actions_ordering(self):
- """
- Ensure that actions are ordered as expected.
- Refs #15964.
- """
- response = self.client.get('/test_admin/admin/admin_views/externalsubscriber/')
- self.assertContains(response, '''<label>Action: <select name="action">
- <option value="" selected="selected">---------</option>
- <option value="delete_selected">Delete selected external
- subscribers</option>
- <option value="redirect_to">Redirect to (Awesome action)</option>
- <option value="external_mail">External mail (Another awesome
- action)</option>
- <option value="download">Download subscription</option>
- <option value="no_perm">No permission to run</option>
- </select>''', html=True)
- def test_model_without_action(self):
- "Tests a ModelAdmin without any action"
- response = self.client.get('/test_admin/admin/admin_views/oldsubscriber/')
- self.assertEqual(response.context["action_form"], None)
- self.assertNotContains(response, '<input type="checkbox" class="action-select"',
- msg_prefix="Found an unexpected action toggle checkboxbox in response")
- self.assertNotContains(response, '<input type="checkbox" class="action-select"')
- def test_model_without_action_still_has_jquery(self):
- "Tests that a ModelAdmin without any actions still gets jQuery included in page"
- response = self.client.get('/test_admin/admin/admin_views/oldsubscriber/')
- self.assertEqual(response.context["action_form"], None)
- self.assertContains(response, 'jquery.min.js',
- msg_prefix="jQuery missing from admin pages for model with no admin actions"
- )
- def test_action_column_class(self):
- "Tests that the checkbox column class is present in the response"
- response = self.client.get('/test_admin/admin/admin_views/subscriber/')
- self.assertNotEqual(response.context["action_form"], None)
- self.assertContains(response, 'action-checkbox-column')
- def test_multiple_actions_form(self):
- """
- Test that actions come from the form whose submit button was pressed (#10618).
- """
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- # Two different actions selected on the two forms...
- 'action': ['external_mail', 'delete_selected'],
- # ...but we clicked "go" on the top form.
- 'index': 0
- }
- response = self.client.post('/test_admin/admin/admin_views/externalsubscriber/', action_data)
- # Send mail, don't delete.
- self.assertEqual(len(mail.outbox), 1)
- self.assertEqual(mail.outbox[0].subject, 'Greetings from a function action')
- def test_user_message_on_none_selected(self):
- """
- User should see a warning when 'Go' is pressed and no items are selected.
- """
- action_data = {
- ACTION_CHECKBOX_NAME: [],
- 'action': 'delete_selected',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/subscriber/', action_data)
- msg = """Items must be selected in order to perform actions on them. No items have been changed."""
- self.assertContains(response, msg)
- self.assertEqual(Subscriber.objects.count(), 2)
- def test_user_message_on_no_action(self):
- """
- User should see a warning when 'Go' is pressed and no action is selected.
- """
- action_data = {
- ACTION_CHECKBOX_NAME: [1, 2],
- 'action': '',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/subscriber/', action_data)
- msg = """No action selected."""
- self.assertContains(response, msg)
- self.assertEqual(Subscriber.objects.count(), 2)
- def test_selection_counter(self):
- """
- Check if the selection counter is there.
- """
- response = self.client.get('/test_admin/admin/admin_views/subscriber/')
- self.assertContains(response, '0 of 2 selected')
- def test_popup_actions(self):
- """ Actions should not be shown in popups. """
- response = self.client.get('/test_admin/admin/admin_views/subscriber/')
- self.assertNotEqual(response.context["action_form"], None)
- response = self.client.get(
- '/test_admin/admin/admin_views/subscriber/?%s' % IS_POPUP_VAR)
- self.assertEqual(response.context["action_form"], None)
- def test_popup_template_response(self):
- """
- Success on popups shall be rendered from template in order to allow
- easy customization.
- """
- response = self.client.post(
- '/test_admin/admin/admin_views/actor/add/?%s=1' % IS_POPUP_VAR,
- {'name': 'Troy McClure', 'age': '55', IS_POPUP_VAR: '1'})
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.template_name, 'admin/popup_response.html')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class TestCustomChangeList(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- urlbit = 'admin'
- def setUp(self):
- result = self.client.login(username='super', password='secret')
- self.assertEqual(result, True)
- def tearDown(self):
- self.client.logout()
- def test_custom_changelist(self):
- """
- Validate that a custom ChangeList class can be used (#9749)
- """
- # Insert some data
- post_data = {"name": "First Gadget"}
- response = self.client.post('/test_admin/%s/admin_views/gadget/add/' % self.urlbit, post_data)
- self.assertEqual(response.status_code, 302) # redirect somewhere
- # Hit the page once to get messages out of the queue message list
- response = self.client.get('/test_admin/%s/admin_views/gadget/' % self.urlbit)
- # Ensure that data is still not visible on the page
- response = self.client.get('/test_admin/%s/admin_views/gadget/' % self.urlbit)
- self.assertEqual(response.status_code, 200)
- self.assertNotContains(response, 'First Gadget')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class TestInlineNotEditable(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- result = self.client.login(username='super', password='secret')
- self.assertEqual(result, True)
- def tearDown(self):
- self.client.logout()
- def test(self):
- """
- InlineModelAdmin broken?
- """
- response = self.client.get('/test_admin/admin/admin_views/parent/add/')
- self.assertEqual(response.status_code, 200)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminCustomQuerysetTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- self.pks = [EmptyModel.objects.create().id for i in range(3)]
- def test_changelist_view(self):
- response = self.client.get('/test_admin/admin/admin_views/emptymodel/')
- for i in self.pks:
- if i > 1:
- self.assertContains(response, 'Primary key = %s' % i)
- else:
- self.assertNotContains(response, 'Primary key = %s' % i)
- def test_changelist_view_count_queries(self):
- # create 2 Person objects
- Person.objects.create(name='person1', gender=1)
- Person.objects.create(name='person2', gender=2)
- # 4 queries are expected: 1 for the session, 1 for the user,
- # 1 for the count and 1 for the objects on the page
- with self.assertNumQueries(4):
- resp = self.client.get('/test_admin/admin/admin_views/person/')
- self.assertEqual(resp.context['selection_note'], '0 of 2 selected')
- self.assertEqual(resp.context['selection_note_all'], 'All 2 selected')
- with self.assertNumQueries(4):
- extra = {'q': 'not_in_name'}
- resp = self.client.get('/test_admin/admin/admin_views/person/', extra)
- self.assertEqual(resp.context['selection_note'], '0 of 0 selected')
- self.assertEqual(resp.context['selection_note_all'], 'All 0 selected')
- with self.assertNumQueries(4):
- extra = {'q': 'person'}
- resp = self.client.get('/test_admin/admin/admin_views/person/', extra)
- self.assertEqual(resp.context['selection_note'], '0 of 2 selected')
- self.assertEqual(resp.context['selection_note_all'], 'All 2 selected')
- # here one more count(*) query will run, because filters were applied
- with self.assertNumQueries(5):
- extra = {'gender__exact': '1'}
- resp = self.client.get('/test_admin/admin/admin_views/person/', extra)
- self.assertEqual(resp.context['selection_note'], '0 of 1 selected')
- self.assertEqual(resp.context['selection_note_all'], '1 selected')
- def test_change_view(self):
- for i in self.pks:
- response = self.client.get('/test_admin/admin/admin_views/emptymodel/%s/' % i)
- if i > 1:
- self.assertEqual(response.status_code, 200)
- else:
- self.assertEqual(response.status_code, 404)
- def test_add_model_modeladmin_defer_qs(self):
- # Test for #14529. defer() is used in ModelAdmin.get_queryset()
- # model has __unicode__ method
- self.assertEqual(CoverLetter.objects.count(), 0)
- # Emulate model instance creation via the admin
- post_data = {
- "author": "Candidate, Best",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/coverletter/add/',
- post_data, follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertEqual(CoverLetter.objects.count(), 1)
- # Message should contain non-ugly model verbose name
- self.assertContains(
- response,
- '<li class="success">The cover letter "Candidate, Best" was added successfully.</li>',
- html=True
- )
- # model has no __unicode__ method
- self.assertEqual(ShortMessage.objects.count(), 0)
- # Emulate model instance creation via the admin
- post_data = {
- "content": "What's this SMS thing?",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/shortmessage/add/',
- post_data, follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertEqual(ShortMessage.objects.count(), 1)
- # Message should contain non-ugly model verbose name
- self.assertContains(
- response,
- '<li class="success">The short message "ShortMessage object" was added successfully.</li>',
- html=True
- )
- def test_add_model_modeladmin_only_qs(self):
- # Test for #14529. only() is used in ModelAdmin.get_queryset()
- # model has __unicode__ method
- self.assertEqual(Telegram.objects.count(), 0)
- # Emulate model instance creation via the admin
- post_data = {
- "title": "Urgent telegram",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/telegram/add/',
- post_data, follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertEqual(Telegram.objects.count(), 1)
- # Message should contain non-ugly model verbose name
- self.assertContains(
- response,
- '<li class="success">The telegram "Urgent telegram" was added successfully.</li>',
- html=True
- )
- # model has no __unicode__ method
- self.assertEqual(Paper.objects.count(), 0)
- # Emulate model instance creation via the admin
- post_data = {
- "title": "My Modified Paper Title",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/paper/add/',
- post_data, follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertEqual(Paper.objects.count(), 1)
- # Message should contain non-ugly model verbose name
- self.assertContains(
- response,
- '<li class="success">The paper "Paper object" was added successfully.</li>',
- html=True
- )
- def test_edit_model_modeladmin_defer_qs(self):
- # Test for #14529. defer() is used in ModelAdmin.get_queryset()
- # model has __unicode__ method
- cl = CoverLetter.objects.create(author="John Doe")
- self.assertEqual(CoverLetter.objects.count(), 1)
- response = self.client.get('/test_admin/admin/admin_views/coverletter/%s/' % cl.pk)
- self.assertEqual(response.status_code, 200)
- # Emulate model instance edit via the admin
- post_data = {
- "author": "John Doe II",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/coverletter/%s/' % cl.pk,
- post_data, follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertEqual(CoverLetter.objects.count(), 1)
- # Message should contain non-ugly model verbose name. Instance
- # representation is set by model's __unicode__()
- self.assertContains(
- response,
- '<li class="success">The cover letter "John Doe II" was changed successfully.</li>',
- html=True
- )
- # model has no __unicode__ method
- sm = ShortMessage.objects.create(content="This is expensive")
- self.assertEqual(ShortMessage.objects.count(), 1)
- response = self.client.get('/test_admin/admin/admin_views/shortmessage/%s/' % sm.pk)
- self.assertEqual(response.status_code, 200)
- # Emulate model instance edit via the admin
- post_data = {
- "content": "Too expensive",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/shortmessage/%s/' % sm.pk,
- post_data, follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertEqual(ShortMessage.objects.count(), 1)
- # Message should contain non-ugly model verbose name. The ugly(!)
- # instance representation is set by six.text_type()
- self.assertContains(
- response,
- '<li class="success">The short message "ShortMessage_Deferred_timestamp object" was changed successfully.</li>',
- html=True
- )
- def test_edit_model_modeladmin_only_qs(self):
- # Test for #14529. only() is used in ModelAdmin.get_queryset()
- # model has __unicode__ method
- t = Telegram.objects.create(title="Frist Telegram")
- self.assertEqual(Telegram.objects.count(), 1)
- response = self.client.get('/test_admin/admin/admin_views/telegram/%s/' % t.pk)
- self.assertEqual(response.status_code, 200)
- # Emulate model instance edit via the admin
- post_data = {
- "title": "Telegram without typo",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/telegram/%s/' % t.pk,
- post_data, follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertEqual(Telegram.objects.count(), 1)
- # Message should contain non-ugly model verbose name. The instance
- # representation is set by model's __unicode__()
- self.assertContains(
- response,
- '<li class="success">The telegram "Telegram without typo" was changed successfully.</li>',
- html=True
- )
- # model has no __unicode__ method
- p = Paper.objects.create(title="My Paper Title")
- self.assertEqual(Paper.objects.count(), 1)
- response = self.client.get('/test_admin/admin/admin_views/paper/%s/' % p.pk)
- self.assertEqual(response.status_code, 200)
- # Emulate model instance edit via the admin
- post_data = {
- "title": "My Modified Paper Title",
- "_save": "Save",
- }
- response = self.client.post('/test_admin/admin/admin_views/paper/%s/' % p.pk,
- post_data, follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertEqual(Paper.objects.count(), 1)
- # Message should contain non-ugly model verbose name. The ugly(!)
- # instance representation is set by six.text_type()
- self.assertContains(
- response,
- '<li class="success">The paper "Paper_Deferred_author object" was changed successfully.</li>',
- html=True
- )
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminInlineFileUploadTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml', 'admin-views-actions.xml']
- urlbit = 'admin'
- def setUp(self):
- self.client.login(username='super', password='secret')
- # Set up test Picture and Gallery.
- # These must be set up here instead of in fixtures in order to allow Picture
- # to use a NamedTemporaryFile.
- tdir = tempfile.gettempdir()
- file1 = tempfile.NamedTemporaryFile(suffix=".file1", dir=tdir)
- file1.write(b'a' * (2 ** 21))
- filename = file1.name
- file1.close()
- self.gallery = Gallery(name="Test Gallery")
- self.gallery.save()
- self.picture = Picture(name="Test Picture", image=filename, gallery=self.gallery)
- self.picture.save()
- def tearDown(self):
- self.client.logout()
- def test_inline_file_upload_edit_validation_error_post(self):
- """
- Test that inline file uploads correctly display prior data (#10002).
- """
- post_data = {
- "name": "Test Gallery",
- "pictures-TOTAL_FORMS": "2",
- "pictures-INITIAL_FORMS": "1",
- "pictures-MAX_NUM_FORMS": "0",
- "pictures-0-id": six.text_type(self.picture.id),
- "pictures-0-gallery": six.text_type(self.gallery.id),
- "pictures-0-name": "Test Picture",
- "pictures-0-image": "",
- "pictures-1-id": "",
- "pictures-1-gallery": str(self.gallery.id),
- "pictures-1-name": "Test Picture 2",
- "pictures-1-image": "",
- }
- response = self.client.post('/test_admin/%s/admin_views/gallery/%d/' % (self.urlbit, self.gallery.id), post_data)
- self.assertTrue(response._container[0].find("Currently:") > -1)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminInlineTests(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.post_data = {
- "name": "Test Name",
- "widget_set-TOTAL_FORMS": "3",
- "widget_set-INITIAL_FORMS": "0",
- "widget_set-MAX_NUM_FORMS": "0",
- "widget_set-0-id": "",
- "widget_set-0-owner": "1",
- "widget_set-0-name": "",
- "widget_set-1-id": "",
- "widget_set-1-owner": "1",
- "widget_set-1-name": "",
- "widget_set-2-id": "",
- "widget_set-2-owner": "1",
- "widget_set-2-name": "",
- "doohickey_set-TOTAL_FORMS": "3",
- "doohickey_set-INITIAL_FORMS": "0",
- "doohickey_set-MAX_NUM_FORMS": "0",
- "doohickey_set-0-owner": "1",
- "doohickey_set-0-code": "",
- "doohickey_set-0-name": "",
- "doohickey_set-1-owner": "1",
- "doohickey_set-1-code": "",
- "doohickey_set-1-name": "",
- "doohickey_set-2-owner": "1",
- "doohickey_set-2-code": "",
- "doohickey_set-2-name": "",
- "grommet_set-TOTAL_FORMS": "3",
- "grommet_set-INITIAL_FORMS": "0",
- "grommet_set-MAX_NUM_FORMS": "0",
- "grommet_set-0-code": "",
- "grommet_set-0-owner": "1",
- "grommet_set-0-name": "",
- "grommet_set-1-code": "",
- "grommet_set-1-owner": "1",
- "grommet_set-1-name": "",
- "grommet_set-2-code": "",
- "grommet_set-2-owner": "1",
- "grommet_set-2-name": "",
- "whatsit_set-TOTAL_FORMS": "3",
- "whatsit_set-INITIAL_FORMS": "0",
- "whatsit_set-MAX_NUM_FORMS": "0",
- "whatsit_set-0-owner": "1",
- "whatsit_set-0-index": "",
- "whatsit_set-0-name": "",
- "whatsit_set-1-owner": "1",
- "whatsit_set-1-index": "",
- "whatsit_set-1-name": "",
- "whatsit_set-2-owner": "1",
- "whatsit_set-2-index": "",
- "whatsit_set-2-name": "",
- "fancydoodad_set-TOTAL_FORMS": "3",
- "fancydoodad_set-INITIAL_FORMS": "0",
- "fancydoodad_set-MAX_NUM_FORMS": "0",
- "fancydoodad_set-0-doodad_ptr": "",
- "fancydoodad_set-0-owner": "1",
- "fancydoodad_set-0-name": "",
- "fancydoodad_set-0-expensive": "on",
- "fancydoodad_set-1-doodad_ptr": "",
- "fancydoodad_set-1-owner": "1",
- "fancydoodad_set-1-name": "",
- "fancydoodad_set-1-expensive": "on",
- "fancydoodad_set-2-doodad_ptr": "",
- "fancydoodad_set-2-owner": "1",
- "fancydoodad_set-2-name": "",
- "fancydoodad_set-2-expensive": "on",
- "category_set-TOTAL_FORMS": "3",
- "category_set-INITIAL_FORMS": "0",
- "category_set-MAX_NUM_FORMS": "0",
- "category_set-0-order": "",
- "category_set-0-id": "",
- "category_set-0-collector": "1",
- "category_set-1-order": "",
- "category_set-1-id": "",
- "category_set-1-collector": "1",
- "category_set-2-order": "",
- "category_set-2-id": "",
- "category_set-2-collector": "1",
- }
- result = self.client.login(username='super', password='secret')
- self.assertEqual(result, True)
- self.collector = Collector(pk=1, name='John Fowles')
- self.collector.save()
- def tearDown(self):
- self.client.logout()
- def test_simple_inline(self):
- "A simple model can be saved as inlines"
- # First add a new inline
- self.post_data['widget_set-0-name'] = "Widget 1"
- collector_url = '/test_admin/admin/admin_views/collector/%d/' % self.collector.pk
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Widget.objects.count(), 1)
- self.assertEqual(Widget.objects.all()[0].name, "Widget 1")
- widget_id = Widget.objects.all()[0].id
- # Check that the PK link exists on the rendered form
- response = self.client.get(collector_url)
- self.assertContains(response, 'name="widget_set-0-id"')
- # Now resave that inline
- self.post_data['widget_set-INITIAL_FORMS'] = "1"
- self.post_data['widget_set-0-id'] = str(widget_id)
- self.post_data['widget_set-0-name'] = "Widget 1"
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Widget.objects.count(), 1)
- self.assertEqual(Widget.objects.all()[0].name, "Widget 1")
- # Now modify that inline
- self.post_data['widget_set-INITIAL_FORMS'] = "1"
- self.post_data['widget_set-0-id'] = str(widget_id)
- self.post_data['widget_set-0-name'] = "Widget 1 Updated"
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Widget.objects.count(), 1)
- self.assertEqual(Widget.objects.all()[0].name, "Widget 1 Updated")
- def test_explicit_autofield_inline(self):
- "A model with an explicit autofield primary key can be saved as inlines. Regression for #8093"
- # First add a new inline
- self.post_data['grommet_set-0-name'] = "Grommet 1"
- collector_url = '/test_admin/admin/admin_views/collector/%d/' % self.collector.pk
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Grommet.objects.count(), 1)
- self.assertEqual(Grommet.objects.all()[0].name, "Grommet 1")
- # Check that the PK link exists on the rendered form
- response = self.client.get(collector_url)
- self.assertContains(response, 'name="grommet_set-0-code"')
- # Now resave that inline
- self.post_data['grommet_set-INITIAL_FORMS'] = "1"
- self.post_data['grommet_set-0-code'] = str(Grommet.objects.all()[0].code)
- self.post_data['grommet_set-0-name'] = "Grommet 1"
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Grommet.objects.count(), 1)
- self.assertEqual(Grommet.objects.all()[0].name, "Grommet 1")
- # Now modify that inline
- self.post_data['grommet_set-INITIAL_FORMS'] = "1"
- self.post_data['grommet_set-0-code'] = str(Grommet.objects.all()[0].code)
- self.post_data['grommet_set-0-name'] = "Grommet 1 Updated"
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Grommet.objects.count(), 1)
- self.assertEqual(Grommet.objects.all()[0].name, "Grommet 1 Updated")
- def test_char_pk_inline(self):
- "A model with a character PK can be saved as inlines. Regression for #10992"
- # First add a new inline
- self.post_data['doohickey_set-0-code'] = "DH1"
- self.post_data['doohickey_set-0-name'] = "Doohickey 1"
- collector_url = '/test_admin/admin/admin_views/collector/%d/' % self.collector.pk
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(DooHickey.objects.count(), 1)
- self.assertEqual(DooHickey.objects.all()[0].name, "Doohickey 1")
- # Check that the PK link exists on the rendered form
- response = self.client.get(collector_url)
- self.assertContains(response, 'name="doohickey_set-0-code"')
- # Now resave that inline
- self.post_data['doohickey_set-INITIAL_FORMS'] = "1"
- self.post_data['doohickey_set-0-code'] = "DH1"
- self.post_data['doohickey_set-0-name'] = "Doohickey 1"
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(DooHickey.objects.count(), 1)
- self.assertEqual(DooHickey.objects.all()[0].name, "Doohickey 1")
- # Now modify that inline
- self.post_data['doohickey_set-INITIAL_FORMS'] = "1"
- self.post_data['doohickey_set-0-code'] = "DH1"
- self.post_data['doohickey_set-0-name'] = "Doohickey 1 Updated"
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(DooHickey.objects.count(), 1)
- self.assertEqual(DooHickey.objects.all()[0].name, "Doohickey 1 Updated")
- def test_integer_pk_inline(self):
- "A model with an integer PK can be saved as inlines. Regression for #10992"
- # First add a new inline
- self.post_data['whatsit_set-0-index'] = "42"
- self.post_data['whatsit_set-0-name'] = "Whatsit 1"
- response = self.client.post('/test_admin/admin/admin_views/collector/1/', self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Whatsit.objects.count(), 1)
- self.assertEqual(Whatsit.objects.all()[0].name, "Whatsit 1")
- # Check that the PK link exists on the rendered form
- response = self.client.get('/test_admin/admin/admin_views/collector/1/')
- self.assertContains(response, 'name="whatsit_set-0-index"')
- # Now resave that inline
- self.post_data['whatsit_set-INITIAL_FORMS'] = "1"
- self.post_data['whatsit_set-0-index'] = "42"
- self.post_data['whatsit_set-0-name'] = "Whatsit 1"
- response = self.client.post('/test_admin/admin/admin_views/collector/1/', self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Whatsit.objects.count(), 1)
- self.assertEqual(Whatsit.objects.all()[0].name, "Whatsit 1")
- # Now modify that inline
- self.post_data['whatsit_set-INITIAL_FORMS'] = "1"
- self.post_data['whatsit_set-0-index'] = "42"
- self.post_data['whatsit_set-0-name'] = "Whatsit 1 Updated"
- response = self.client.post('/test_admin/admin/admin_views/collector/1/', self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Whatsit.objects.count(), 1)
- self.assertEqual(Whatsit.objects.all()[0].name, "Whatsit 1 Updated")
- def test_inherited_inline(self):
- "An inherited model can be saved as inlines. Regression for #11042"
- # First add a new inline
- self.post_data['fancydoodad_set-0-name'] = "Fancy Doodad 1"
- collector_url = '/test_admin/admin/admin_views/collector/%d/' % self.collector.pk
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(FancyDoodad.objects.count(), 1)
- self.assertEqual(FancyDoodad.objects.all()[0].name, "Fancy Doodad 1")
- doodad_pk = FancyDoodad.objects.all()[0].pk
- # Check that the PK link exists on the rendered form
- response = self.client.get(collector_url)
- self.assertContains(response, 'name="fancydoodad_set-0-doodad_ptr"')
- # Now resave that inline
- self.post_data['fancydoodad_set-INITIAL_FORMS'] = "1"
- self.post_data['fancydoodad_set-0-doodad_ptr'] = str(doodad_pk)
- self.post_data['fancydoodad_set-0-name'] = "Fancy Doodad 1"
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(FancyDoodad.objects.count(), 1)
- self.assertEqual(FancyDoodad.objects.all()[0].name, "Fancy Doodad 1")
- # Now modify that inline
- self.post_data['fancydoodad_set-INITIAL_FORMS'] = "1"
- self.post_data['fancydoodad_set-0-doodad_ptr'] = str(doodad_pk)
- self.post_data['fancydoodad_set-0-name'] = "Fancy Doodad 1 Updated"
- response = self.client.post(collector_url, self.post_data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(FancyDoodad.objects.count(), 1)
- self.assertEqual(FancyDoodad.objects.all()[0].name, "Fancy Doodad 1 Updated")
- def test_ordered_inline(self):
- """Check that an inline with an editable ordering fields is
- updated correctly. Regression for #10922"""
- # Create some objects with an initial ordering
- Category.objects.create(id=1, order=1, collector=self.collector)
- Category.objects.create(id=2, order=2, collector=self.collector)
- Category.objects.create(id=3, order=0, collector=self.collector)
- Category.objects.create(id=4, order=0, collector=self.collector)
- # NB: The order values must be changed so that the items are reordered.
- self.post_data.update({
- "name": "Frederick Clegg",
- "category_set-TOTAL_FORMS": "7",
- "category_set-INITIAL_FORMS": "4",
- "category_set-MAX_NUM_FORMS": "0",
- "category_set-0-order": "14",
- "category_set-0-id": "1",
- "category_set-0-collector": "1",
- "category_set-1-order": "13",
- "category_set-1-id": "2",
- "category_set-1-collector": "1",
- "category_set-2-order": "1",
- "category_set-2-id": "3",
- "category_set-2-collector": "1",
- "category_set-3-order": "0",
- "category_set-3-id": "4",
- "category_set-3-collector": "1",
- "category_set-4-order": "",
- "category_set-4-id": "",
- "category_set-4-collector": "1",
- "category_set-5-order": "",
- "category_set-5-id": "",
- "category_set-5-collector": "1",
- "category_set-6-order": "",
- "category_set-6-id": "",
- "category_set-6-collector": "1",
- })
- response = self.client.post('/test_admin/admin/admin_views/collector/1/', self.post_data)
- # Successful post will redirect
- self.assertEqual(response.status_code, 302)
- # Check that the order values have been applied to the right objects
- self.assertEqual(self.collector.category_set.count(), 4)
- self.assertEqual(Category.objects.get(id=1).order, 14)
- self.assertEqual(Category.objects.get(id=2).order, 13)
- self.assertEqual(Category.objects.get(id=3).order, 1)
- self.assertEqual(Category.objects.get(id=4).order, 0)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class NeverCacheTests(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml', 'admin-views-colors.xml', 'admin-views-fabrics.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def testAdminIndex(self):
- "Check the never-cache status of the main index"
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(get_max_age(response), 0)
- def testAppIndex(self):
- "Check the never-cache status of an application index"
- response = self.client.get('/test_admin/admin/admin_views/')
- self.assertEqual(get_max_age(response), 0)
- def testModelIndex(self):
- "Check the never-cache status of a model index"
- response = self.client.get('/test_admin/admin/admin_views/fabric/')
- self.assertEqual(get_max_age(response), 0)
- def testModelAdd(self):
- "Check the never-cache status of a model add page"
- response = self.client.get('/test_admin/admin/admin_views/fabric/add/')
- self.assertEqual(get_max_age(response), 0)
- def testModelView(self):
- "Check the never-cache status of a model edit page"
- response = self.client.get('/test_admin/admin/admin_views/section/1/')
- self.assertEqual(get_max_age(response), 0)
- def testModelHistory(self):
- "Check the never-cache status of a model history page"
- response = self.client.get('/test_admin/admin/admin_views/section/1/history/')
- self.assertEqual(get_max_age(response), 0)
- def testModelDelete(self):
- "Check the never-cache status of a model delete page"
- response = self.client.get('/test_admin/admin/admin_views/section/1/delete/')
- self.assertEqual(get_max_age(response), 0)
- def testLogin(self):
- "Check the never-cache status of login views"
- self.client.logout()
- response = self.client.get('/test_admin/admin/')
- self.assertEqual(get_max_age(response), 0)
- def testLogout(self):
- "Check the never-cache status of logout view"
- response = self.client.get('/test_admin/admin/logout/')
- self.assertEqual(get_max_age(response), 0)
- def testPasswordChange(self):
- "Check the never-cache status of the password change view"
- self.client.logout()
- response = self.client.get('/test_admin/password_change/')
- self.assertEqual(get_max_age(response), None)
- def testPasswordChangeDone(self):
- "Check the never-cache status of the password change done view"
- response = self.client.get('/test_admin/admin/password_change/done/')
- self.assertEqual(get_max_age(response), None)
- def testJsi18n(self):
- "Check the never-cache status of the JavaScript i18n view"
- response = self.client.get('/test_admin/admin/jsi18n/')
- self.assertEqual(get_max_age(response), None)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class PrePopulatedTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_prepopulated_on(self):
- response = self.client.get('/test_admin/admin/admin_views/prepopulatedpost/add/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, "id: '#id_slug',")
- self.assertContains(response, "field['dependency_ids'].push('#id_title');")
- self.assertContains(response, "id: '#id_prepopulatedsubpost_set-0-subslug',")
- def test_prepopulated_off(self):
- response = self.client.get('/test_admin/admin/admin_views/prepopulatedpost/1/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, "A Long Title")
- self.assertNotContains(response, "id: '#id_slug'")
- self.assertNotContains(response, "field['dependency_ids'].push('#id_title');")
- self.assertNotContains(response, "id: '#id_prepopulatedsubpost_set-0-subslug',")
- @override_settings(USE_THOUSAND_SEPARATOR=True, USE_L10N=True)
- def test_prepopulated_maxlength_localized(self):
- """
- Regression test for #15938: if USE_THOUSAND_SEPARATOR is set, make sure
- that maxLength (in the JavaScript) is rendered without separators.
- """
- response = self.client.get('/test_admin/admin/admin_views/prepopulatedpostlargeslug/add/')
- self.assertContains(response, "maxLength: 1000") # instead of 1,000
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class SeleniumAdminViewsFirefoxTests(AdminSeleniumWebDriverTestCase):
- available_apps = ['admin_views'] + AdminSeleniumWebDriverTestCase.available_apps
- fixtures = ['admin-views-users.xml']
- urls = "admin_views.urls"
- webdriver_class = 'selenium.webdriver.firefox.webdriver.WebDriver'
- def test_prepopulated_fields(self):
- """
- Ensure that the JavaScript-automated prepopulated fields work with the
- main form and with stacked and tabular inlines.
- Refs #13068, #9264, #9983, #9784.
- """
- from selenium.common.exceptions import TimeoutException
- self.admin_login(username='super', password='secret', login_url='/test_admin/admin/')
- self.selenium.get('%s%s' % (self.live_server_url,
- '/test_admin/admin/admin_views/mainprepopulated/add/'))
- # Main form ----------------------------------------------------------
- self.selenium.find_element_by_css_selector('#id_pubdate').send_keys('2012-02-18')
- self.get_select_option('#id_status', 'option two').click()
- self.selenium.find_element_by_css_selector('#id_name').send_keys(' this is the mAin nÀMë and it\'s awεšome')
- slug1 = self.selenium.find_element_by_css_selector('#id_slug1').get_attribute('value')
- slug2 = self.selenium.find_element_by_css_selector('#id_slug2').get_attribute('value')
- self.assertEqual(slug1, 'main-name-and-its-awesome-2012-02-18')
- self.assertEqual(slug2, 'option-two-main-name-and-its-awesome')
- # Stacked inlines ----------------------------------------------------
- # Initial inline
- self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-0-pubdate').send_keys('2011-12-17')
- self.get_select_option('#id_relatedprepopulated_set-0-status', 'option one').click()
- self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-0-name').send_keys(' here is a sŤāÇkeð inline ! ')
- slug1 = self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-0-slug1').get_attribute('value')
- slug2 = self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-0-slug2').get_attribute('value')
- self.assertEqual(slug1, 'here-stacked-inline-2011-12-17')
- self.assertEqual(slug2, 'option-one-here-stacked-inline')
- # Add an inline
- self.selenium.find_elements_by_link_text('Add another Related Prepopulated')[0].click()
- self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-1-pubdate').send_keys('1999-01-25')
- self.get_select_option('#id_relatedprepopulated_set-1-status', 'option two').click()
- self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-1-name').send_keys(' now you haVe anöther sŤāÇkeð inline with a very ... loooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooog text... ')
- slug1 = self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-1-slug1').get_attribute('value')
- slug2 = self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-1-slug2').get_attribute('value')
- self.assertEqual(slug1, 'now-you-have-another-stacked-inline-very-loooooooo') # 50 characters maximum for slug1 field
- self.assertEqual(slug2, 'option-two-now-you-have-another-stacked-inline-very-looooooo') # 60 characters maximum for slug2 field
- # Tabular inlines ----------------------------------------------------
- # Initial inline
- self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-2-0-pubdate').send_keys('1234-12-07')
- self.get_select_option('#id_relatedprepopulated_set-2-0-status', 'option two').click()
- self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-2-0-name').send_keys('And now, with a tÃbűlaŘ inline !!!')
- slug1 = self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-2-0-slug1').get_attribute('value')
- slug2 = self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-2-0-slug2').get_attribute('value')
- self.assertEqual(slug1, 'and-now-tabular-inline-1234-12-07')
- self.assertEqual(slug2, 'option-two-and-now-tabular-inline')
- # Add an inline
- self.selenium.find_elements_by_link_text('Add another Related Prepopulated')[1].click()
- self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-2-1-pubdate').send_keys('1981-08-22')
- self.get_select_option('#id_relatedprepopulated_set-2-1-status', 'option one').click()
- self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-2-1-name').send_keys('a tÃbűlaŘ inline with ignored ;"&*^\%$#@-/`~ characters')
- slug1 = self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-2-1-slug1').get_attribute('value')
- slug2 = self.selenium.find_element_by_css_selector('#id_relatedprepopulated_set-2-1-slug2').get_attribute('value')
- self.assertEqual(slug1, 'tabular-inline-ignored-characters-1981-08-22')
- self.assertEqual(slug2, 'option-one-tabular-inline-ignored-characters')
- # Save and check that everything is properly stored in the database
- self.selenium.find_element_by_xpath('//input[@value="Save"]').click()
- self.wait_page_loaded()
- self.assertEqual(MainPrepopulated.objects.all().count(), 1)
- MainPrepopulated.objects.get(
- name=' this is the mAin nÀMë and it\'s awεšome',
- pubdate='2012-02-18',
- status='option two',
- slug1='main-name-and-its-awesome-2012-02-18',
- slug2='option-two-main-name-and-its-awesome',
- )
- self.assertEqual(RelatedPrepopulated.objects.all().count(), 4)
- RelatedPrepopulated.objects.get(
- name=' here is a sŤāÇkeð inline ! ',
- pubdate='2011-12-17',
- status='option one',
- slug1='here-stacked-inline-2011-12-17',
- slug2='option-one-here-stacked-inline',
- )
- RelatedPrepopulated.objects.get(
- name=' now you haVe anöther sŤāÇkeð inline with a very ... loooooooooooooooooo', # 75 characters in name field
- pubdate='1999-01-25',
- status='option two',
- slug1='now-you-have-another-stacked-inline-very-loooooooo',
- slug2='option-two-now-you-have-another-stacked-inline-very-looooooo',
- )
- RelatedPrepopulated.objects.get(
- name='And now, with a tÃbűlaŘ inline !!!',
- pubdate='1234-12-07',
- status='option two',
- slug1='and-now-tabular-inline-1234-12-07',
- slug2='option-two-and-now-tabular-inline',
- )
- RelatedPrepopulated.objects.get(
- name='a tÃbűlaŘ inline with ignored ;"&*^\%$#@-/`~ characters',
- pubdate='1981-08-22',
- status='option one',
- slug1='tabular-inline-ignored-characters-1981-08-22',
- slug2='option-one-tabular-inline-ignored-characters',
- )
- def test_populate_existing_object(self):
- """
- Ensure that the prepopulation works for existing objects too, as long
- as the original field is empty.
- Refs #19082.
- """
- # Slugs are empty to start with.
- item = MainPrepopulated.objects.create(
- name=' this is the mAin nÀMë',
- pubdate='2012-02-18',
- status='option two',
- slug1='',
- slug2='',
- )
- self.admin_login(username='super',
- password='secret',
- login_url='/test_admin/admin/')
- object_url = '%s%s' % (
- self.live_server_url,
- '/test_admin/admin/admin_views/mainprepopulated/{}/'.format(item.id))
- self.selenium.get(object_url)
- self.selenium.find_element_by_css_selector('#id_name').send_keys(' the best')
- # The slugs got prepopulated since they were originally empty
- slug1 = self.selenium.find_element_by_css_selector('#id_slug1').get_attribute('value')
- slug2 = self.selenium.find_element_by_css_selector('#id_slug2').get_attribute('value')
- self.assertEqual(slug1, 'main-name-best-2012-02-18')
- self.assertEqual(slug2, 'option-two-main-name-best')
- # Save the object
- self.selenium.find_element_by_xpath('//input[@value="Save"]').click()
- self.wait_page_loaded()
- self.selenium.get(object_url)
- self.selenium.find_element_by_css_selector('#id_name').send_keys(' hello')
- # The slugs got prepopulated didn't change since they were originally not empty
- slug1 = self.selenium.find_element_by_css_selector('#id_slug1').get_attribute('value')
- slug2 = self.selenium.find_element_by_css_selector('#id_slug2').get_attribute('value')
- self.assertEqual(slug1, 'main-name-best-2012-02-18')
- self.assertEqual(slug2, 'option-two-main-name-best')
- def test_collapsible_fieldset(self):
- """
- Test that the 'collapse' class in fieldsets definition allows to
- show/hide the appropriate field section.
- """
- self.admin_login(username='super', password='secret', login_url='/test_admin/admin/')
- self.selenium.get('%s%s' % (self.live_server_url,
- '/test_admin/admin/admin_views/article/add/'))
- self.assertFalse(self.selenium.find_element_by_id('id_title').is_displayed())
- self.selenium.find_elements_by_link_text('Show')[0].click()
- self.assertTrue(self.selenium.find_element_by_id('id_title').is_displayed())
- self.assertEqual(
- self.selenium.find_element_by_id('fieldsetcollapser0').text,
- "Hide"
- )
- class SeleniumAdminViewsChromeTests(SeleniumAdminViewsFirefoxTests):
- webdriver_class = 'selenium.webdriver.chrome.webdriver.WebDriver'
- class SeleniumAdminViewsIETests(SeleniumAdminViewsFirefoxTests):
- webdriver_class = 'selenium.webdriver.ie.webdriver.WebDriver'
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class ReadonlyTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_readonly_get(self):
- response = self.client.get('/test_admin/admin/admin_views/post/add/')
- self.assertEqual(response.status_code, 200)
- self.assertNotContains(response, 'name="posted"')
- # 3 fields + 2 submit buttons + 4 inline management form fields, + 2
- # hidden fields for inlines + 1 field for the inline + 2 empty form
- self.assertContains(response, "<input", count=14)
- self.assertContains(response, formats.localize(datetime.date.today()))
- self.assertContains(response,
- "<label>Awesomeness level:</label>")
- self.assertContains(response, "Very awesome.")
- self.assertContains(response, "Unkown coolness.")
- self.assertContains(response, "foo")
- # Checks that multiline text in a readonly field gets <br /> tags
- self.assertContains(response, "Multiline<br />test<br />string")
- self.assertContains(response, "InlineMultiline<br />test<br />string")
- self.assertContains(response,
- formats.localize(datetime.date.today() - datetime.timedelta(days=7))
- )
- self.assertContains(response, '<div class="form-row field-coolness">')
- self.assertContains(response, '<div class="form-row field-awesomeness_level">')
- self.assertContains(response, '<div class="form-row field-posted">')
- self.assertContains(response, '<div class="form-row field-value">')
- self.assertContains(response, '<div class="form-row">')
- self.assertContains(response, '<p class="help">', 3)
- self.assertContains(response, '<p class="help">Some help text for the title (with unicode ŠĐĆŽćžšđ)</p>', html=True)
- self.assertContains(response, '<p class="help">Some help text for the content (with unicode ŠĐĆŽćžšđ)</p>', html=True)
- self.assertContains(response, '<p class="help">Some help text for the date (with unicode ŠĐĆŽćžšđ)</p>', html=True)
- p = Post.objects.create(title="I worked on readonly_fields", content="Its good stuff")
- response = self.client.get('/test_admin/admin/admin_views/post/%d/' % p.pk)
- self.assertContains(response, "%d amount of cool" % p.pk)
- def test_readonly_post(self):
- data = {
- "title": "Django Got Readonly Fields",
- "content": "This is an incredible development.",
- "link_set-TOTAL_FORMS": "1",
- "link_set-INITIAL_FORMS": "0",
- "link_set-MAX_NUM_FORMS": "0",
- }
- response = self.client.post('/test_admin/admin/admin_views/post/add/', data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Post.objects.count(), 1)
- p = Post.objects.get()
- self.assertEqual(p.posted, datetime.date.today())
- data["posted"] = "10-8-1990" # some date that's not today
- response = self.client.post('/test_admin/admin/admin_views/post/add/', data)
- self.assertEqual(response.status_code, 302)
- self.assertEqual(Post.objects.count(), 2)
- p = Post.objects.order_by('-id')[0]
- self.assertEqual(p.posted, datetime.date.today())
- def test_readonly_manytomany(self):
- "Regression test for #13004"
- response = self.client.get('/test_admin/admin/admin_views/pizza/add/')
- self.assertEqual(response.status_code, 200)
- def test_user_password_change_limited_queryset(self):
- su = User.objects.filter(is_superuser=True)[0]
- response = self.client.get('/test_admin/admin2/auth/user/%s/password/' % su.pk)
- self.assertEqual(response.status_code, 404)
- def test_change_form_renders_correct_null_choice_value(self):
- """
- Regression test for #17911.
- """
- choice = Choice.objects.create(choice=None)
- response = self.client.get('/test_admin/admin/admin_views/choice/%s/' % choice.pk)
- self.assertContains(response, '<p>No opinion</p>', html=True)
- self.assertNotContains(response, '<p>(None)</p>')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class RawIdFieldsTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_limit_choices_to(self):
- """Regression test for 14880"""
- actor = Actor.objects.create(name="Palin", age=27)
- inquisition1 = Inquisition.objects.create(expected=True,
- leader=actor,
- country="England")
- inquisition2 = Inquisition.objects.create(expected=False,
- leader=actor,
- country="Spain")
- response = self.client.get('/test_admin/admin/admin_views/sketch/add/')
- # Find the link
- m = re.search(br'<a href="([^"]*)"[^>]* id="lookup_id_inquisition"', response.content)
- self.assertTrue(m) # Got a match
- popup_url = m.groups()[0].decode().replace("&", "&")
- # Handle relative links
- popup_url = urljoin(response.request['PATH_INFO'], popup_url)
- # Get the popup and verify the correct objects show up in the resulting
- # page. This step also tests integers, strings and booleans in the
- # lookup query string; in model we define inquisition field to have a
- # limit_choices_to option that includes a filter on a string field
- # (inquisition__actor__name), a filter on an integer field
- # (inquisition__actor__age), and a filter on a boolean field
- # (inquisition__expected).
- response2 = self.client.get(popup_url)
- self.assertContains(response2, "Spain")
- self.assertNotContains(response2, "England")
- def test_limit_choices_to_isnull_false(self):
- """Regression test for 20182"""
- Actor.objects.create(name="Palin", age=27)
- Actor.objects.create(name="Kilbraken", age=50, title="Judge")
- response = self.client.get('/test_admin/admin/admin_views/sketch/add/')
- # Find the link
- m = re.search(br'<a href="([^"]*)"[^>]* id="lookup_id_defendant0"', response.content)
- self.assertTrue(m) # Got a match
- popup_url = m.groups()[0].decode().replace("&", "&")
- # Handle relative links
- popup_url = urljoin(response.request['PATH_INFO'], popup_url)
- # Get the popup and verify the correct objects show up in the resulting
- # page. This step tests field__isnull=0 gets parsed correctly from the
- # lookup query string; in model we define defendant0 field to have a
- # limit_choices_to option that includes "actor__title__isnull=False".
- response2 = self.client.get(popup_url)
- self.assertContains(response2, "Kilbraken")
- self.assertNotContains(response2, "Palin")
- def test_limit_choices_to_isnull_true(self):
- """Regression test for 20182"""
- Actor.objects.create(name="Palin", age=27)
- Actor.objects.create(name="Kilbraken", age=50, title="Judge")
- response = self.client.get('/test_admin/admin/admin_views/sketch/add/')
- # Find the link
- m = re.search(br'<a href="([^"]*)"[^>]* id="lookup_id_defendant1"', response.content)
- self.assertTrue(m) # Got a match
- popup_url = m.groups()[0].decode().replace("&", "&")
- # Handle relative links
- popup_url = urljoin(response.request['PATH_INFO'], popup_url)
- # Get the popup and verify the correct objects show up in the resulting
- # page. This step tests field__isnull=1 gets parsed correctly from the
- # lookup query string; in model we define defendant1 field to have a
- # limit_choices_to option that includes "actor__title__isnull=True".
- response2 = self.client.get(popup_url)
- self.assertNotContains(response2, "Kilbraken")
- self.assertContains(response2, "Palin")
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class UserAdminTest(TestCase):
- """
- Tests user CRUD functionality.
- """
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_save_button(self):
- user_count = User.objects.count()
- response = self.client.post('/test_admin/admin/auth/user/add/', {
- 'username': 'newuser',
- 'password1': 'newpassword',
- 'password2': 'newpassword',
- })
- new_user = User.objects.order_by('-id')[0]
- self.assertRedirects(response, '/test_admin/admin/auth/user/%s/' % new_user.pk)
- self.assertEqual(User.objects.count(), user_count + 1)
- self.assertTrue(new_user.has_usable_password())
- def test_save_continue_editing_button(self):
- user_count = User.objects.count()
- response = self.client.post('/test_admin/admin/auth/user/add/', {
- 'username': 'newuser',
- 'password1': 'newpassword',
- 'password2': 'newpassword',
- '_continue': '1',
- })
- new_user = User.objects.order_by('-id')[0]
- self.assertRedirects(response, '/test_admin/admin/auth/user/%s/' % new_user.pk)
- self.assertEqual(User.objects.count(), user_count + 1)
- self.assertTrue(new_user.has_usable_password())
- def test_password_mismatch(self):
- response = self.client.post('/test_admin/admin/auth/user/add/', {
- 'username': 'newuser',
- 'password1': 'newpassword',
- 'password2': 'mismatch',
- })
- self.assertEqual(response.status_code, 200)
- adminform = response.context['adminform']
- self.assertTrue('password' not in adminform.form.errors)
- self.assertEqual(adminform.form.errors['password2'],
- ["The two password fields didn't match."])
- def test_user_fk_popup(self):
- """Quick user addition in a FK popup shouldn't invoke view for further user customization"""
- response = self.client.get('/test_admin/admin/admin_views/album/add/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, '/test_admin/admin/auth/user/add')
- self.assertContains(response, 'class="add-another" id="add_id_owner" onclick="return showAddAnotherPopup(this);"')
- response = self.client.get('/test_admin/admin/auth/user/add/?_popup=1')
- self.assertEqual(response.status_code, 200)
- self.assertNotContains(response, 'name="_continue"')
- self.assertNotContains(response, 'name="_addanother"')
- data = {
- 'username': 'newuser',
- 'password1': 'newpassword',
- 'password2': 'newpassword',
- '_popup': '1',
- '_save': '1',
- }
- response = self.client.post('/test_admin/admin/auth/user/add/?_popup=1', data, follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'dismissAddAnotherPopup')
- def test_save_add_another_button(self):
- user_count = User.objects.count()
- response = self.client.post('/test_admin/admin/auth/user/add/', {
- 'username': 'newuser',
- 'password1': 'newpassword',
- 'password2': 'newpassword',
- '_addanother': '1',
- })
- new_user = User.objects.order_by('-id')[0]
- self.assertRedirects(response, '/test_admin/admin/auth/user/add/')
- self.assertEqual(User.objects.count(), user_count + 1)
- self.assertTrue(new_user.has_usable_password())
- def test_user_permission_performance(self):
- u = User.objects.all()[0]
- # Don't depend on a warm cache, see #17377.
- ContentType.objects.clear_cache()
- expected_queries = 10
- # Oracle doesn't implement "RELEASE SAVPOINT", see #20387.
- if connection.vendor == 'oracle':
- expected_queries -= 1
- with self.assertNumQueries(expected_queries):
- response = self.client.get('/test_admin/admin/auth/user/%s/' % u.pk)
- self.assertEqual(response.status_code, 200)
- def test_form_url_present_in_context(self):
- u = User.objects.all()[0]
- response = self.client.get('/test_admin/admin3/auth/user/%s/password/' % u.pk)
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.context['form_url'], 'pony')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class GroupAdminTest(TestCase):
- """
- Tests group CRUD functionality.
- """
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_save_button(self):
- group_count = Group.objects.count()
- response = self.client.post('/test_admin/admin/auth/group/add/', {
- 'name': 'newgroup',
- })
- new_group = Group.objects.order_by('-id')[0]
- self.assertRedirects(response, '/test_admin/admin/auth/group/')
- self.assertEqual(Group.objects.count(), group_count + 1)
- def test_group_permission_performance(self):
- g = Group.objects.create(name="test_group")
- expected_queries = 8
- # Oracle doesn't implement "RELEASE SAVPOINT", see #20387.
- if connection.vendor == 'oracle':
- expected_queries -= 1
- with self.assertNumQueries(expected_queries):
- response = self.client.get('/test_admin/admin/auth/group/%s/' % g.pk)
- self.assertEqual(response.status_code, 200)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class CSSTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_field_prefix_css_classes(self):
- """
- Ensure that fields have a CSS class name with a 'field-' prefix.
- Refs #16371.
- """
- response = self.client.get('/test_admin/admin/admin_views/post/add/')
- # The main form
- self.assertContains(response, 'class="form-row field-title"')
- self.assertContains(response, 'class="form-row field-content"')
- self.assertContains(response, 'class="form-row field-public"')
- self.assertContains(response, 'class="form-row field-awesomeness_level"')
- self.assertContains(response, 'class="form-row field-coolness"')
- self.assertContains(response, 'class="form-row field-value"')
- self.assertContains(response, 'class="form-row"') # The lambda function
- # The tabular inline
- self.assertContains(response, '<td class="field-url">')
- self.assertContains(response, '<td class="field-posted">')
- def test_index_css_classes(self):
- """
- Ensure that CSS class names are used for each app and model on the
- admin index pages.
- Refs #17050.
- """
- # General index page
- response = self.client.get("/test_admin/admin/")
- self.assertContains(response, '<div class="app-admin_views module">')
- self.assertContains(response, '<tr class="model-actor">')
- self.assertContains(response, '<tr class="model-album">')
- # App index page
- response = self.client.get("/test_admin/admin/admin_views/")
- self.assertContains(response, '<div class="app-admin_views module">')
- self.assertContains(response, '<tr class="model-actor">')
- self.assertContains(response, '<tr class="model-album">')
- def testAppModelInFormBodyClass(self):
- """
- Ensure app and model tag are correcly read by change_form template
- """
- response = self.client.get('/test_admin/admin/admin_views/section/add/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response,
- '<body class="app-admin_views model-section ')
- def testAppModelInListBodyClass(self):
- """
- Ensure app and model tag are correcly read by change_list template
- """
- response = self.client.get('/test_admin/admin/admin_views/section/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response,
- '<body class="app-admin_views model-section ')
- def testAppModelInDeleteConfirmationBodyClass(self):
- """
- Ensure app and model tag are correcly read by delete_confirmation
- template
- """
- response = self.client.get(
- '/test_admin/admin/admin_views/section/1/delete/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response,
- '<body class="app-admin_views model-section ')
- def testAppModelInAppIndexBodyClass(self):
- """
- Ensure app and model tag are correcly read by app_index template
- """
- response = self.client.get('/test_admin/admin/admin_views/')
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, '<body class="app-admin_views ')
- def testAppModelInDeleteSelectedConfirmationBodyClass(self):
- """
- Ensure app and model tag are correcly read by
- delete_selected_confirmation template
- """
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- 'action': 'delete_selected',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/section/',
- action_data)
- self.assertEqual(response.status_code, 200)
- self.assertContains(response,
- '<body class="app-admin_views model-section ')
- def test_changelist_field_classes(self):
- """
- Cells of the change list table should contain the field name in their class attribute
- Refs #11195.
- """
- Podcast.objects.create(name="Django Dose",
- release_date=datetime.date.today())
- response = self.client.get('/test_admin/admin/admin_views/podcast/')
- self.assertContains(
- response, '<th class="field-name">')
- self.assertContains(
- response, '<td class="field-release_date nowrap">')
- self.assertContains(
- response, '<td class="action-checkbox">')
- try:
- import docutils
- except ImportError:
- docutils = None
- @unittest.skipUnless(docutils, "no docutils installed.")
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminDocsTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_tags(self):
- response = self.client.get('/test_admin/admin/doc/tags/')
- # The builtin tag group exists
- self.assertContains(response, "<h2>Built-in tags</h2>", count=2, html=True)
- # A builtin tag exists in both the index and detail
- self.assertContains(response, '<h3 id="built_in-autoescape">autoescape</h3>', html=True)
- self.assertContains(response, '<li><a href="#built_in-autoescape">autoescape</a></li>', html=True)
- # An app tag exists in both the index and detail
- self.assertContains(response, '<h3 id="flatpages-get_flatpages">get_flatpages</h3>', html=True)
- self.assertContains(response, '<li><a href="#flatpages-get_flatpages">get_flatpages</a></li>', html=True)
- # The admin list tag group exists
- self.assertContains(response, "<h2>admin_list</h2>", count=2, html=True)
- # An admin list tag exists in both the index and detail
- self.assertContains(response, '<h3 id="admin_list-admin_actions">admin_actions</h3>', html=True)
- self.assertContains(response, '<li><a href="#admin_list-admin_actions">admin_actions</a></li>', html=True)
- def test_filters(self):
- response = self.client.get('/test_admin/admin/doc/filters/')
- # The builtin filter group exists
- self.assertContains(response, "<h2>Built-in filters</h2>", count=2, html=True)
- # A builtin filter exists in both the index and detail
- self.assertContains(response, '<h3 id="built_in-add">add</h3>', html=True)
- self.assertContains(response, '<li><a href="#built_in-add">add</a></li>', html=True)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class ValidXHTMLTests(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- urlbit = 'admin'
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- @override_settings(
- TEMPLATE_CONTEXT_PROCESSORS=filter(
- lambda t: t != 'django.core.context_processors.i18n',
- global_settings.TEMPLATE_CONTEXT_PROCESSORS),
- USE_I18N=False,
- )
- def testLangNamePresent(self):
- response = self.client.get('/test_admin/%s/admin_views/' % self.urlbit)
- self.assertNotContains(response, ' lang=""')
- self.assertNotContains(response, ' xml:lang=""')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class DateHierarchyTests(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- self.old_USE_THOUSAND_SEPARATOR = settings.USE_THOUSAND_SEPARATOR
- self.old_USE_L10N = settings.USE_L10N
- settings.USE_THOUSAND_SEPARATOR = True
- settings.USE_L10N = True
- def tearDown(self):
- settings.USE_THOUSAND_SEPARATOR = self.old_USE_THOUSAND_SEPARATOR
- settings.USE_L10N = self.old_USE_L10N
- formats.reset_format_cache()
- def assert_non_localized_year(self, response, year):
- """Ensure that the year is not localized with
- USE_THOUSAND_SEPARATOR. Refs #15234.
- """
- self.assertNotContains(response, formats.number_format(year))
- def assert_contains_year_link(self, response, date):
- self.assertContains(response, '?release_date__year=%d"' % (date.year,))
- def assert_contains_month_link(self, response, date):
- self.assertContains(
- response, '?release_date__month=%d&release_date__year=%d"' % (
- date.month, date.year))
- def assert_contains_day_link(self, response, date):
- self.assertContains(
- response, '?release_date__day=%d&'
- 'release_date__month=%d&release_date__year=%d"' % (
- date.day, date.month, date.year))
- def test_empty(self):
- """
- Ensure that no date hierarchy links display with empty changelist.
- """
- response = self.client.get(
- reverse('admin:admin_views_podcast_changelist'))
- self.assertNotContains(response, 'release_date__year=')
- self.assertNotContains(response, 'release_date__month=')
- self.assertNotContains(response, 'release_date__day=')
- def test_single(self):
- """
- Ensure that single day-level date hierarchy appears for single object.
- """
- DATE = datetime.date(2000, 6, 30)
- Podcast.objects.create(release_date=DATE)
- url = reverse('admin:admin_views_podcast_changelist')
- response = self.client.get(url)
- self.assert_contains_day_link(response, DATE)
- self.assert_non_localized_year(response, 2000)
- def test_within_month(self):
- """
- Ensure that day-level links appear for changelist within single month.
- """
- DATES = (datetime.date(2000, 6, 30),
- datetime.date(2000, 6, 15),
- datetime.date(2000, 6, 3))
- for date in DATES:
- Podcast.objects.create(release_date=date)
- url = reverse('admin:admin_views_podcast_changelist')
- response = self.client.get(url)
- for date in DATES:
- self.assert_contains_day_link(response, date)
- self.assert_non_localized_year(response, 2000)
- def test_within_year(self):
- """
- Ensure that month-level links appear for changelist within single year.
- """
- DATES = (datetime.date(2000, 1, 30),
- datetime.date(2000, 3, 15),
- datetime.date(2000, 5, 3))
- for date in DATES:
- Podcast.objects.create(release_date=date)
- url = reverse('admin:admin_views_podcast_changelist')
- response = self.client.get(url)
- # no day-level links
- self.assertNotContains(response, 'release_date__day=')
- for date in DATES:
- self.assert_contains_month_link(response, date)
- self.assert_non_localized_year(response, 2000)
- def test_multiple_years(self):
- """
- Ensure that year-level links appear for year-spanning changelist.
- """
- DATES = (datetime.date(2001, 1, 30),
- datetime.date(2003, 3, 15),
- datetime.date(2005, 5, 3))
- for date in DATES:
- Podcast.objects.create(release_date=date)
- response = self.client.get(
- reverse('admin:admin_views_podcast_changelist'))
- # no day/month-level links
- self.assertNotContains(response, 'release_date__day=')
- self.assertNotContains(response, 'release_date__month=')
- for date in DATES:
- self.assert_contains_year_link(response, date)
- # and make sure GET parameters still behave correctly
- for date in DATES:
- url = '%s?release_date__year=%d' % (
- reverse('admin:admin_views_podcast_changelist'),
- date.year)
- response = self.client.get(url)
- self.assert_contains_month_link(response, date)
- self.assert_non_localized_year(response, 2000)
- self.assert_non_localized_year(response, 2003)
- self.assert_non_localized_year(response, 2005)
- url = '%s?release_date__year=%d&release_date__month=%d' % (
- reverse('admin:admin_views_podcast_changelist'),
- date.year, date.month)
- response = self.client.get(url)
- self.assert_contains_day_link(response, date)
- self.assert_non_localized_year(response, 2000)
- self.assert_non_localized_year(response, 2003)
- self.assert_non_localized_year(response, 2005)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminCustomSaveRelatedTests(TestCase):
- """
- Ensure that one can easily customize the way related objects are saved.
- Refs #16115.
- """
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def test_should_be_able_to_edit_related_objects_on_add_view(self):
- post = {
- 'child_set-TOTAL_FORMS': '3',
- 'child_set-INITIAL_FORMS': '0',
- 'name': 'Josh Stone',
- 'child_set-0-name': 'Paul',
- 'child_set-1-name': 'Catherine',
- }
- response = self.client.post('/test_admin/admin/admin_views/parent/add/', post)
- self.assertEqual(1, Parent.objects.count())
- self.assertEqual(2, Child.objects.count())
- children_names = list(Child.objects.order_by('name').values_list('name', flat=True))
- self.assertEqual('Josh Stone', Parent.objects.latest('id').name)
- self.assertEqual(['Catherine Stone', 'Paul Stone'], children_names)
- def test_should_be_able_to_edit_related_objects_on_change_view(self):
- parent = Parent.objects.create(name='Josh Stone')
- paul = Child.objects.create(parent=parent, name='Paul')
- catherine = Child.objects.create(parent=parent, name='Catherine')
- post = {
- 'child_set-TOTAL_FORMS': '5',
- 'child_set-INITIAL_FORMS': '2',
- 'name': 'Josh Stone',
- 'child_set-0-name': 'Paul',
- 'child_set-0-id': paul.id,
- 'child_set-1-name': 'Catherine',
- 'child_set-1-id': catherine.id,
- }
- response = self.client.post('/test_admin/admin/admin_views/parent/%s/' % parent.id, post)
- children_names = list(Child.objects.order_by('name').values_list('name', flat=True))
- self.assertEqual('Josh Stone', Parent.objects.latest('id').name)
- self.assertEqual(['Catherine Stone', 'Paul Stone'], children_names)
- def test_should_be_able_to_edit_related_objects_on_changelist_view(self):
- parent = Parent.objects.create(name='Josh Rock')
- paul = Child.objects.create(parent=parent, name='Paul')
- catherine = Child.objects.create(parent=parent, name='Catherine')
- post = {
- 'form-TOTAL_FORMS': '1',
- 'form-INITIAL_FORMS': '1',
- 'form-MAX_NUM_FORMS': '0',
- 'form-0-id': parent.id,
- 'form-0-name': 'Josh Stone',
- '_save': 'Save'
- }
- response = self.client.post('/test_admin/admin/admin_views/parent/', post)
- children_names = list(Child.objects.order_by('name').values_list('name', flat=True))
- self.assertEqual('Josh Stone', Parent.objects.latest('id').name)
- self.assertEqual(['Catherine Stone', 'Paul Stone'], children_names)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminViewLogoutTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def test_client_logout_url_can_be_used_to_login(self):
- response = self.client.get('/test_admin/admin/logout/')
- self.assertEqual(response.status_code, 200)
- self.assertTemplateUsed(response, 'registration/logged_out.html')
- self.assertEqual(response.request['PATH_INFO'], '/test_admin/admin/logout/')
- # we are now logged out
- response = self.client.get('/test_admin/admin/logout/')
- self.assertEqual(response.status_code, 302) # we should be redirected to the login page.
- # follow the redirect and test results.
- response = self.client.get('/test_admin/admin/logout/', follow=True)
- self.assertEqual(response.status_code, 200)
- self.assertTemplateUsed(response, 'admin/login.html')
- self.assertEqual(response.request['PATH_INFO'], '/test_admin/admin/')
- self.assertContains(response, '<input type="hidden" name="next" value="/test_admin/admin/" />')
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminUserMessageTest(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def send_message(self, level):
- """
- Helper that sends a post to the dummy test methods and asserts that a
- message with the level has appeared in the response.
- """
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- 'action': 'message_%s' % level,
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/usermessenger/',
- action_data, follow=True)
- self.assertContains(response,
- '<li class="%s">Test %s</li>' % (level, level),
- html=True)
- @override_settings(MESSAGE_LEVEL=10) # Set to DEBUG for this request
- def test_message_debug(self):
- self.send_message('debug')
- def test_message_info(self):
- self.send_message('info')
- def test_message_success(self):
- self.send_message('success')
- def test_message_warning(self):
- self.send_message('warning')
- def test_message_error(self):
- self.send_message('error')
- def test_message_extra_tags(self):
- action_data = {
- ACTION_CHECKBOX_NAME: [1],
- 'action': 'message_extra_tags',
- 'index': 0,
- }
- response = self.client.post('/test_admin/admin/admin_views/usermessenger/',
- action_data, follow=True)
- self.assertContains(response,
- '<li class="extra_tag info">Test tags</li>',
- html=True)
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class AdminKeepChangeListFiltersTests(TestCase):
- urls = "admin_views.urls"
- fixtures = ['admin-views-users.xml']
- admin_site = site
- def setUp(self):
- self.client.login(username='super', password='secret')
- def tearDown(self):
- self.client.logout()
- def get_changelist_filters_querystring(self):
- return urlencode({
- 'is_superuser__exact': 0,
- 'is_staff__exact': 0,
- })
- def get_preserved_filters_querystring(self):
- return urlencode({
- '_changelist_filters': self.get_changelist_filters_querystring()
- })
- def get_sample_user_id(self):
- return 104
- def get_changelist_url(self):
- return '%s?%s' % (
- reverse('admin:auth_user_changelist',
- current_app=self.admin_site.name),
- self.get_changelist_filters_querystring(),
- )
- def get_add_url(self):
- return '%s?%s' % (
- reverse('admin:auth_user_add',
- current_app=self.admin_site.name),
- self.get_preserved_filters_querystring(),
- )
- def get_change_url(self, user_id=None):
- if user_id is None:
- user_id = self.get_sample_user_id()
- return "%s?%s" % (
- reverse('admin:auth_user_change', args=(user_id,),
- current_app=self.admin_site.name),
- self.get_preserved_filters_querystring(),
- )
- def get_history_url(self, user_id=None):
- if user_id is None:
- user_id = self.get_sample_user_id()
- return "%s?%s" % (
- reverse('admin:auth_user_history', args=(user_id,),
- current_app=self.admin_site.name),
- self.get_preserved_filters_querystring(),
- )
- def get_delete_url(self, user_id=None):
- if user_id is None:
- user_id = self.get_sample_user_id()
- return "%s?%s" % (
- reverse('admin:auth_user_delete', args=(user_id,),
- current_app=self.admin_site.name),
- self.get_preserved_filters_querystring(),
- )
- def test_changelist_view(self):
- response = self.client.get(self.get_changelist_url())
- self.assertEqual(response.status_code, 200)
- # Check the `change_view` link has the correct querystring.
- detail_link = """<a href="%s">joepublic</a>""" % self.get_change_url()
- self.assertContains(response, detail_link, count=1)
- def test_change_view(self):
- # Get the `change_view`.
- response = self.client.get(self.get_change_url())
- self.assertEqual(response.status_code, 200)
- # Check the form action.
- form_action = """<form enctype="multipart/form-data" action="?%s" method="post" id="user_form">""" % self.get_preserved_filters_querystring()
- self.assertContains(response, form_action, count=1)
- # Check the history link.
- history_link = """<a href="%s" class="historylink">History</a>""" % self.get_history_url()
- self.assertContains(response, history_link, count=1)
- # Check the delete link.
- delete_link = """<a href="%s" class="deletelink">Delete</a>""" % (self.get_delete_url())
- self.assertContains(response, delete_link, count=1)
- # Test redirect on "Save".
- post_data = {
- 'username': 'joepublic',
- 'last_login_0': '2007-05-30',
- 'last_login_1': '13:20:10',
- 'date_joined_0': '2007-05-30',
- 'date_joined_1': '13:20:10',
- }
- post_data['_save'] = 1
- response = self.client.post(self.get_change_url(), data=post_data)
- self.assertRedirects(response, self.get_changelist_url())
- post_data.pop('_save')
- # Test redirect on "Save and continue".
- post_data['_continue'] = 1
- response = self.client.post(self.get_change_url(), data=post_data)
- self.assertRedirects(response, self.get_change_url())
- post_data.pop('_continue')
- # Test redirect on "Save and add new".
- post_data['_addanother'] = 1
- response = self.client.post(self.get_change_url(), data=post_data)
- self.assertRedirects(response, self.get_add_url())
- post_data.pop('_addanother')
- def test_add_view(self):
- # Get the `add_view`.
- response = self.client.get(self.get_add_url())
- self.assertEqual(response.status_code, 200)
- # Check the form action.
- form_action = """<form enctype="multipart/form-data" action="?%s" method="post" id="user_form">""" % self.get_preserved_filters_querystring()
- self.assertContains(response, form_action, count=1)
- # Test redirect on "Save".
- post_data = {
- 'username': 'dummy',
- 'password1': 'test',
- 'password2': 'test',
- }
- post_data['_save'] = 1
- response = self.client.post(self.get_add_url(), data=post_data)
- self.assertRedirects(response, self.get_change_url(User.objects.latest('pk').pk))
- post_data.pop('_save')
- # Test redirect on "Save and continue".
- post_data['username'] = 'dummy2'
- post_data['_continue'] = 1
- response = self.client.post(self.get_add_url(), data=post_data)
- self.assertRedirects(response, self.get_change_url(User.objects.latest('pk').pk))
- post_data.pop('_continue')
- # Test redirect on "Save and add new".
- post_data['username'] = 'dummy3'
- post_data['_addanother'] = 1
- response = self.client.post(self.get_add_url(), data=post_data)
- self.assertRedirects(response, self.get_add_url())
- post_data.pop('_addanother')
- def test_delete_view(self):
- # Test redirect on "Delete".
- response = self.client.post(self.get_delete_url(), {'post': 'yes'})
- self.assertRedirects(response, self.get_changelist_url())
- class NamespacedAdminKeepChangeListFiltersTests(AdminKeepChangeListFiltersTests):
- admin_site = site2
|