1
0

config_test.py 177 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428
  1. # encoding: utf-8
  2. from __future__ import absolute_import
  3. from __future__ import print_function
  4. from __future__ import unicode_literals
  5. import codecs
  6. import os
  7. import shutil
  8. import tempfile
  9. from operator import itemgetter
  10. from random import shuffle
  11. import py
  12. import pytest
  13. import yaml
  14. from ...helpers import build_config_details
  15. from ...helpers import BUSYBOX_IMAGE_WITH_TAG
  16. from compose.config import config
  17. from compose.config import types
  18. from compose.config.config import ConfigFile
  19. from compose.config.config import resolve_build_args
  20. from compose.config.config import resolve_environment
  21. from compose.config.environment import Environment
  22. from compose.config.errors import ConfigurationError
  23. from compose.config.errors import VERSION_EXPLANATION
  24. from compose.config.serialize import denormalize_service_dict
  25. from compose.config.serialize import serialize_config
  26. from compose.config.serialize import serialize_ns_time_value
  27. from compose.config.types import VolumeSpec
  28. from compose.const import COMPOSEFILE_V1 as V1
  29. from compose.const import COMPOSEFILE_V2_0 as V2_0
  30. from compose.const import COMPOSEFILE_V2_1 as V2_1
  31. from compose.const import COMPOSEFILE_V2_2 as V2_2
  32. from compose.const import COMPOSEFILE_V2_3 as V2_3
  33. from compose.const import COMPOSEFILE_V3_0 as V3_0
  34. from compose.const import COMPOSEFILE_V3_1 as V3_1
  35. from compose.const import COMPOSEFILE_V3_2 as V3_2
  36. from compose.const import COMPOSEFILE_V3_3 as V3_3
  37. from compose.const import COMPOSEFILE_V3_5 as V3_5
  38. from compose.const import IS_WINDOWS_PLATFORM
  39. from tests import mock
  40. from tests import unittest
  41. DEFAULT_VERSION = V2_0
  42. def make_service_dict(name, service_dict, working_dir='.', filename=None):
  43. """Test helper function to construct a ServiceExtendsResolver
  44. """
  45. resolver = config.ServiceExtendsResolver(
  46. config.ServiceConfig(
  47. working_dir=working_dir,
  48. filename=filename,
  49. name=name,
  50. config=service_dict),
  51. config.ConfigFile(filename=filename, config={}),
  52. environment=Environment.from_env_file(working_dir)
  53. )
  54. return config.process_service(resolver.run())
  55. def service_sort(services):
  56. return sorted(services, key=itemgetter('name'))
  57. def secret_sort(secrets):
  58. return sorted(secrets, key=itemgetter('source'))
  59. class ConfigTest(unittest.TestCase):
  60. def test_load(self):
  61. service_dicts = config.load(
  62. build_config_details(
  63. {
  64. 'foo': {'image': 'busybox'},
  65. 'bar': {'image': 'busybox', 'environment': ['FOO=1']},
  66. },
  67. 'tests/fixtures/extends',
  68. 'common.yml'
  69. )
  70. ).services
  71. assert service_sort(service_dicts) == service_sort([
  72. {
  73. 'name': 'bar',
  74. 'image': 'busybox',
  75. 'environment': {'FOO': '1'},
  76. },
  77. {
  78. 'name': 'foo',
  79. 'image': 'busybox',
  80. }
  81. ])
  82. def test_load_v2(self):
  83. config_data = config.load(
  84. build_config_details({
  85. 'version': '2',
  86. 'services': {
  87. 'foo': {'image': 'busybox'},
  88. 'bar': {'image': 'busybox', 'environment': ['FOO=1']},
  89. },
  90. 'volumes': {
  91. 'hello': {
  92. 'driver': 'default',
  93. 'driver_opts': {'beep': 'boop'}
  94. }
  95. },
  96. 'networks': {
  97. 'default': {
  98. 'driver': 'bridge',
  99. 'driver_opts': {'beep': 'boop'}
  100. },
  101. 'with_ipam': {
  102. 'ipam': {
  103. 'driver': 'default',
  104. 'config': [
  105. {'subnet': '172.28.0.0/16'}
  106. ]
  107. }
  108. },
  109. 'internal': {
  110. 'driver': 'bridge',
  111. 'internal': True
  112. }
  113. }
  114. }, 'working_dir', 'filename.yml')
  115. )
  116. service_dicts = config_data.services
  117. volume_dict = config_data.volumes
  118. networks_dict = config_data.networks
  119. assert service_sort(service_dicts) == service_sort([
  120. {
  121. 'name': 'bar',
  122. 'image': 'busybox',
  123. 'environment': {'FOO': '1'},
  124. },
  125. {
  126. 'name': 'foo',
  127. 'image': 'busybox',
  128. }
  129. ])
  130. assert volume_dict == {
  131. 'hello': {
  132. 'driver': 'default',
  133. 'driver_opts': {'beep': 'boop'}
  134. }
  135. }
  136. assert networks_dict == {
  137. 'default': {
  138. 'driver': 'bridge',
  139. 'driver_opts': {'beep': 'boop'}
  140. },
  141. 'with_ipam': {
  142. 'ipam': {
  143. 'driver': 'default',
  144. 'config': [
  145. {'subnet': '172.28.0.0/16'}
  146. ]
  147. }
  148. },
  149. 'internal': {
  150. 'driver': 'bridge',
  151. 'internal': True
  152. }
  153. }
  154. def test_valid_versions(self):
  155. for version in ['2', '2.0']:
  156. cfg = config.load(build_config_details({'version': version}))
  157. assert cfg.version == V2_0
  158. cfg = config.load(build_config_details({'version': '2.1'}))
  159. assert cfg.version == V2_1
  160. cfg = config.load(build_config_details({'version': '2.2'}))
  161. assert cfg.version == V2_2
  162. cfg = config.load(build_config_details({'version': '2.3'}))
  163. assert cfg.version == V2_3
  164. for version in ['3', '3.0']:
  165. cfg = config.load(build_config_details({'version': version}))
  166. assert cfg.version == V3_0
  167. cfg = config.load(build_config_details({'version': '3.1'}))
  168. assert cfg.version == V3_1
  169. def test_v1_file_version(self):
  170. cfg = config.load(build_config_details({'web': {'image': 'busybox'}}))
  171. assert cfg.version == V1
  172. assert list(s['name'] for s in cfg.services) == ['web']
  173. cfg = config.load(build_config_details({'version': {'image': 'busybox'}}))
  174. assert cfg.version == V1
  175. assert list(s['name'] for s in cfg.services) == ['version']
  176. def test_wrong_version_type(self):
  177. for version in [None, 1, 2, 2.0]:
  178. with pytest.raises(ConfigurationError) as excinfo:
  179. config.load(
  180. build_config_details(
  181. {'version': version},
  182. filename='filename.yml',
  183. )
  184. )
  185. assert 'Version in "filename.yml" is invalid - it should be a string.' \
  186. in excinfo.exconly()
  187. def test_unsupported_version(self):
  188. with pytest.raises(ConfigurationError) as excinfo:
  189. config.load(
  190. build_config_details(
  191. {'version': '2.18'},
  192. filename='filename.yml',
  193. )
  194. )
  195. assert 'Version in "filename.yml" is unsupported' in excinfo.exconly()
  196. assert VERSION_EXPLANATION in excinfo.exconly()
  197. def test_version_1_is_invalid(self):
  198. with pytest.raises(ConfigurationError) as excinfo:
  199. config.load(
  200. build_config_details(
  201. {
  202. 'version': '1',
  203. 'web': {'image': 'busybox'},
  204. },
  205. filename='filename.yml',
  206. )
  207. )
  208. assert 'Version in "filename.yml" is invalid' in excinfo.exconly()
  209. assert VERSION_EXPLANATION in excinfo.exconly()
  210. def test_v1_file_with_version_is_invalid(self):
  211. with pytest.raises(ConfigurationError) as excinfo:
  212. config.load(
  213. build_config_details(
  214. {
  215. 'version': '2',
  216. 'web': {'image': 'busybox'},
  217. },
  218. filename='filename.yml',
  219. )
  220. )
  221. assert 'Invalid top-level property "web"' in excinfo.exconly()
  222. assert VERSION_EXPLANATION in excinfo.exconly()
  223. def test_named_volume_config_empty(self):
  224. config_details = build_config_details({
  225. 'version': '2',
  226. 'services': {
  227. 'simple': {'image': 'busybox'}
  228. },
  229. 'volumes': {
  230. 'simple': None,
  231. 'other': {},
  232. }
  233. })
  234. config_result = config.load(config_details)
  235. volumes = config_result.volumes
  236. assert 'simple' in volumes
  237. assert volumes['simple'] == {}
  238. assert volumes['other'] == {}
  239. def test_named_volume_numeric_driver_opt(self):
  240. config_details = build_config_details({
  241. 'version': '2',
  242. 'services': {
  243. 'simple': {'image': 'busybox'}
  244. },
  245. 'volumes': {
  246. 'simple': {'driver_opts': {'size': 42}},
  247. }
  248. })
  249. cfg = config.load(config_details)
  250. assert cfg.volumes['simple']['driver_opts']['size'] == '42'
  251. def test_volume_invalid_driver_opt(self):
  252. config_details = build_config_details({
  253. 'version': '2',
  254. 'services': {
  255. 'simple': {'image': 'busybox'}
  256. },
  257. 'volumes': {
  258. 'simple': {'driver_opts': {'size': True}},
  259. }
  260. })
  261. with pytest.raises(ConfigurationError) as exc:
  262. config.load(config_details)
  263. assert 'driver_opts.size contains an invalid type' in exc.exconly()
  264. def test_named_volume_invalid_type_list(self):
  265. config_details = build_config_details({
  266. 'version': '2',
  267. 'services': {
  268. 'simple': {'image': 'busybox'}
  269. },
  270. 'volumes': []
  271. })
  272. with pytest.raises(ConfigurationError) as exc:
  273. config.load(config_details)
  274. assert "volume must be a mapping, not an array" in exc.exconly()
  275. def test_networks_invalid_type_list(self):
  276. config_details = build_config_details({
  277. 'version': '2',
  278. 'services': {
  279. 'simple': {'image': 'busybox'}
  280. },
  281. 'networks': []
  282. })
  283. with pytest.raises(ConfigurationError) as exc:
  284. config.load(config_details)
  285. assert "network must be a mapping, not an array" in exc.exconly()
  286. def test_load_service_with_name_version(self):
  287. with mock.patch('compose.config.config.log') as mock_logging:
  288. config_data = config.load(
  289. build_config_details({
  290. 'version': {
  291. 'image': 'busybox'
  292. }
  293. }, 'working_dir', 'filename.yml')
  294. )
  295. assert 'Unexpected type for "version" key in "filename.yml"' \
  296. in mock_logging.warning.call_args[0][0]
  297. service_dicts = config_data.services
  298. assert service_sort(service_dicts) == service_sort([
  299. {
  300. 'name': 'version',
  301. 'image': 'busybox',
  302. }
  303. ])
  304. def test_load_throws_error_when_not_dict(self):
  305. with pytest.raises(ConfigurationError):
  306. config.load(
  307. build_config_details(
  308. {'web': BUSYBOX_IMAGE_WITH_TAG},
  309. 'working_dir',
  310. 'filename.yml'
  311. )
  312. )
  313. def test_load_throws_error_when_not_dict_v2(self):
  314. with pytest.raises(ConfigurationError):
  315. config.load(
  316. build_config_details(
  317. {'version': '2', 'services': {'web': BUSYBOX_IMAGE_WITH_TAG}},
  318. 'working_dir',
  319. 'filename.yml'
  320. )
  321. )
  322. def test_load_throws_error_with_invalid_network_fields(self):
  323. with pytest.raises(ConfigurationError):
  324. config.load(
  325. build_config_details({
  326. 'version': '2',
  327. 'services': {'web': BUSYBOX_IMAGE_WITH_TAG},
  328. 'networks': {
  329. 'invalid': {'foo', 'bar'}
  330. }
  331. }, 'working_dir', 'filename.yml')
  332. )
  333. def test_load_config_link_local_ips_network(self):
  334. base_file = config.ConfigFile(
  335. 'base.yaml',
  336. {
  337. 'version': str(V2_1),
  338. 'services': {
  339. 'web': {
  340. 'image': 'example/web',
  341. 'networks': {
  342. 'foobar': {
  343. 'aliases': ['foo', 'bar'],
  344. 'link_local_ips': ['169.254.8.8']
  345. }
  346. }
  347. }
  348. },
  349. 'networks': {'foobar': {}}
  350. }
  351. )
  352. details = config.ConfigDetails('.', [base_file])
  353. web_service = config.load(details).services[0]
  354. assert web_service['networks'] == {
  355. 'foobar': {
  356. 'aliases': ['foo', 'bar'],
  357. 'link_local_ips': ['169.254.8.8']
  358. }
  359. }
  360. def test_load_config_service_labels(self):
  361. base_file = config.ConfigFile(
  362. 'base.yaml',
  363. {
  364. 'version': '2.1',
  365. 'services': {
  366. 'web': {
  367. 'image': 'example/web',
  368. 'labels': ['label_key=label_val']
  369. },
  370. 'db': {
  371. 'image': 'example/db',
  372. 'labels': {
  373. 'label_key': 'label_val'
  374. }
  375. }
  376. },
  377. }
  378. )
  379. details = config.ConfigDetails('.', [base_file])
  380. service_dicts = config.load(details).services
  381. for service in service_dicts:
  382. assert service['labels'] == {
  383. 'label_key': 'label_val'
  384. }
  385. def test_load_config_custom_resource_names(self):
  386. base_file = config.ConfigFile(
  387. 'base.yaml', {
  388. 'version': '3.5',
  389. 'volumes': {
  390. 'abc': {
  391. 'name': 'xyz'
  392. }
  393. },
  394. 'networks': {
  395. 'abc': {
  396. 'name': 'xyz'
  397. }
  398. },
  399. 'secrets': {
  400. 'abc': {
  401. 'name': 'xyz'
  402. }
  403. },
  404. 'configs': {
  405. 'abc': {
  406. 'name': 'xyz'
  407. }
  408. }
  409. }
  410. )
  411. details = config.ConfigDetails('.', [base_file])
  412. loaded_config = config.load(details)
  413. assert loaded_config.networks['abc'] == {'name': 'xyz'}
  414. assert loaded_config.volumes['abc'] == {'name': 'xyz'}
  415. assert loaded_config.secrets['abc']['name'] == 'xyz'
  416. assert loaded_config.configs['abc']['name'] == 'xyz'
  417. def test_load_config_volume_and_network_labels(self):
  418. base_file = config.ConfigFile(
  419. 'base.yaml',
  420. {
  421. 'version': '2.1',
  422. 'services': {
  423. 'web': {
  424. 'image': 'example/web',
  425. },
  426. },
  427. 'networks': {
  428. 'with_label': {
  429. 'labels': {
  430. 'label_key': 'label_val'
  431. }
  432. }
  433. },
  434. 'volumes': {
  435. 'with_label': {
  436. 'labels': {
  437. 'label_key': 'label_val'
  438. }
  439. }
  440. }
  441. }
  442. )
  443. details = config.ConfigDetails('.', [base_file])
  444. loaded_config = config.load(details)
  445. assert loaded_config.networks == {
  446. 'with_label': {
  447. 'labels': {
  448. 'label_key': 'label_val'
  449. }
  450. }
  451. }
  452. assert loaded_config.volumes == {
  453. 'with_label': {
  454. 'labels': {
  455. 'label_key': 'label_val'
  456. }
  457. }
  458. }
  459. def test_load_config_invalid_service_names(self):
  460. for invalid_name in ['?not?allowed', ' ', '', '!', '/', '\xe2']:
  461. with pytest.raises(ConfigurationError) as exc:
  462. config.load(build_config_details(
  463. {invalid_name: {'image': 'busybox'}}))
  464. assert 'Invalid service name \'%s\'' % invalid_name in exc.exconly()
  465. def test_load_config_invalid_service_names_v2(self):
  466. for invalid_name in ['?not?allowed', ' ', '', '!', '/', '\xe2']:
  467. with pytest.raises(ConfigurationError) as exc:
  468. config.load(build_config_details(
  469. {
  470. 'version': '2',
  471. 'services': {invalid_name: {'image': 'busybox'}},
  472. }))
  473. assert 'Invalid service name \'%s\'' % invalid_name in exc.exconly()
  474. def test_load_with_invalid_field_name(self):
  475. with pytest.raises(ConfigurationError) as exc:
  476. config.load(build_config_details(
  477. {
  478. 'version': '2',
  479. 'services': {
  480. 'web': {'image': 'busybox', 'name': 'bogus'},
  481. }
  482. },
  483. 'working_dir',
  484. 'filename.yml',
  485. ))
  486. assert "Unsupported config option for services.web: 'name'" in exc.exconly()
  487. def test_load_with_invalid_field_name_v1(self):
  488. with pytest.raises(ConfigurationError) as exc:
  489. config.load(build_config_details(
  490. {
  491. 'web': {'image': 'busybox', 'name': 'bogus'},
  492. },
  493. 'working_dir',
  494. 'filename.yml',
  495. ))
  496. assert "Unsupported config option for web: 'name'" in exc.exconly()
  497. def test_load_invalid_service_definition(self):
  498. config_details = build_config_details(
  499. {'web': 'wrong'},
  500. 'working_dir',
  501. 'filename.yml')
  502. with pytest.raises(ConfigurationError) as exc:
  503. config.load(config_details)
  504. assert "service 'web' must be a mapping not a string." in exc.exconly()
  505. def test_load_with_empty_build_args(self):
  506. config_details = build_config_details(
  507. {
  508. 'version': '2',
  509. 'services': {
  510. 'web': {
  511. 'build': {
  512. 'context': os.getcwd(),
  513. 'args': None,
  514. },
  515. },
  516. },
  517. }
  518. )
  519. with pytest.raises(ConfigurationError) as exc:
  520. config.load(config_details)
  521. assert (
  522. "services.web.build.args contains an invalid type, it should be an "
  523. "object, or an array" in exc.exconly()
  524. )
  525. def test_config_integer_service_name_raise_validation_error(self):
  526. with pytest.raises(ConfigurationError) as excinfo:
  527. config.load(
  528. build_config_details(
  529. {1: {'image': 'busybox'}},
  530. 'working_dir',
  531. 'filename.yml'
  532. )
  533. )
  534. assert (
  535. "In file 'filename.yml', the service name 1 must be a quoted string, i.e. '1'" in
  536. excinfo.exconly()
  537. )
  538. def test_config_integer_service_name_raise_validation_error_v2(self):
  539. with pytest.raises(ConfigurationError) as excinfo:
  540. config.load(
  541. build_config_details(
  542. {
  543. 'version': '2',
  544. 'services': {1: {'image': 'busybox'}}
  545. },
  546. 'working_dir',
  547. 'filename.yml'
  548. )
  549. )
  550. assert (
  551. "In file 'filename.yml', the service name 1 must be a quoted string, i.e. '1'." in
  552. excinfo.exconly()
  553. )
  554. def test_config_integer_service_name_raise_validation_error_v2_when_no_interpolate(self):
  555. with pytest.raises(ConfigurationError) as excinfo:
  556. config.load(
  557. build_config_details(
  558. {
  559. 'version': '2',
  560. 'services': {1: {'image': 'busybox'}}
  561. },
  562. 'working_dir',
  563. 'filename.yml'
  564. ),
  565. interpolate=False
  566. )
  567. assert (
  568. "In file 'filename.yml', the service name 1 must be a quoted string, i.e. '1'." in
  569. excinfo.exconly()
  570. )
  571. def test_config_integer_service_property_raise_validation_error(self):
  572. with pytest.raises(ConfigurationError) as excinfo:
  573. config.load(
  574. build_config_details({
  575. 'version': '2.1',
  576. 'services': {'foobar': {'image': 'busybox', 1234: 'hah'}}
  577. }, 'working_dir', 'filename.yml')
  578. )
  579. assert (
  580. "Unsupported config option for services.foobar: '1234'" in excinfo.exconly()
  581. )
  582. def test_config_invalid_service_name_raise_validation_error(self):
  583. with pytest.raises(ConfigurationError) as excinfo:
  584. config.load(
  585. build_config_details({
  586. 'version': '2',
  587. 'services': {
  588. 'test_app': {'build': '.'},
  589. 'mong\\o': {'image': 'mongo'},
  590. }
  591. })
  592. )
  593. assert 'Invalid service name \'mong\\o\'' in excinfo.exconly()
  594. def test_config_duplicate_cache_from_values_validation_error(self):
  595. with pytest.raises(ConfigurationError) as exc:
  596. config.load(
  597. build_config_details({
  598. 'version': '2.3',
  599. 'services': {
  600. 'test': {'build': {'context': '.', 'cache_from': ['a', 'b', 'a']}}
  601. }
  602. })
  603. )
  604. assert 'build.cache_from contains non-unique items' in exc.exconly()
  605. def test_load_with_multiple_files_v1(self):
  606. base_file = config.ConfigFile(
  607. 'base.yaml',
  608. {
  609. 'web': {
  610. 'image': 'example/web',
  611. 'links': ['db'],
  612. },
  613. 'db': {
  614. 'image': 'example/db',
  615. },
  616. })
  617. override_file = config.ConfigFile(
  618. 'override.yaml',
  619. {
  620. 'web': {
  621. 'build': '/',
  622. 'volumes': ['/home/user/project:/code'],
  623. },
  624. })
  625. details = config.ConfigDetails('.', [base_file, override_file])
  626. service_dicts = config.load(details).services
  627. expected = [
  628. {
  629. 'name': 'web',
  630. 'build': {'context': os.path.abspath('/')},
  631. 'volumes': [VolumeSpec.parse('/home/user/project:/code')],
  632. 'links': ['db'],
  633. },
  634. {
  635. 'name': 'db',
  636. 'image': 'example/db',
  637. },
  638. ]
  639. assert service_sort(service_dicts) == service_sort(expected)
  640. def test_load_with_multiple_files_and_empty_override(self):
  641. base_file = config.ConfigFile(
  642. 'base.yml',
  643. {'web': {'image': 'example/web'}})
  644. override_file = config.ConfigFile('override.yml', None)
  645. details = config.ConfigDetails('.', [base_file, override_file])
  646. with pytest.raises(ConfigurationError) as exc:
  647. config.load(details)
  648. error_msg = "Top level object in 'override.yml' needs to be an object"
  649. assert error_msg in exc.exconly()
  650. def test_load_with_multiple_files_and_empty_override_v2(self):
  651. base_file = config.ConfigFile(
  652. 'base.yml',
  653. {'version': '2', 'services': {'web': {'image': 'example/web'}}})
  654. override_file = config.ConfigFile('override.yml', None)
  655. details = config.ConfigDetails('.', [base_file, override_file])
  656. with pytest.raises(ConfigurationError) as exc:
  657. config.load(details)
  658. error_msg = "Top level object in 'override.yml' needs to be an object"
  659. assert error_msg in exc.exconly()
  660. def test_load_with_multiple_files_and_empty_base(self):
  661. base_file = config.ConfigFile('base.yml', None)
  662. override_file = config.ConfigFile(
  663. 'override.yml',
  664. {'web': {'image': 'example/web'}})
  665. details = config.ConfigDetails('.', [base_file, override_file])
  666. with pytest.raises(ConfigurationError) as exc:
  667. config.load(details)
  668. assert "Top level object in 'base.yml' needs to be an object" in exc.exconly()
  669. def test_load_with_multiple_files_and_empty_base_v2(self):
  670. base_file = config.ConfigFile('base.yml', None)
  671. override_file = config.ConfigFile(
  672. 'override.tml',
  673. {'version': '2', 'services': {'web': {'image': 'example/web'}}}
  674. )
  675. details = config.ConfigDetails('.', [base_file, override_file])
  676. with pytest.raises(ConfigurationError) as exc:
  677. config.load(details)
  678. assert "Top level object in 'base.yml' needs to be an object" in exc.exconly()
  679. def test_load_with_multiple_files_and_extends_in_override_file(self):
  680. base_file = config.ConfigFile(
  681. 'base.yaml',
  682. {
  683. 'web': {'image': 'example/web'},
  684. })
  685. override_file = config.ConfigFile(
  686. 'override.yaml',
  687. {
  688. 'web': {
  689. 'extends': {
  690. 'file': 'common.yml',
  691. 'service': 'base',
  692. },
  693. 'volumes': ['/home/user/project:/code'],
  694. },
  695. })
  696. details = config.ConfigDetails('.', [base_file, override_file])
  697. tmpdir = py.test.ensuretemp('config_test')
  698. self.addCleanup(tmpdir.remove)
  699. tmpdir.join('common.yml').write("""
  700. base:
  701. labels: ['label=one']
  702. """)
  703. with tmpdir.as_cwd():
  704. service_dicts = config.load(details).services
  705. expected = [
  706. {
  707. 'name': 'web',
  708. 'image': 'example/web',
  709. 'volumes': [VolumeSpec.parse('/home/user/project:/code')],
  710. 'labels': {'label': 'one'},
  711. },
  712. ]
  713. assert service_sort(service_dicts) == service_sort(expected)
  714. def test_load_mixed_extends_resolution(self):
  715. main_file = config.ConfigFile(
  716. 'main.yml', {
  717. 'version': '2.2',
  718. 'services': {
  719. 'prodweb': {
  720. 'extends': {
  721. 'service': 'web',
  722. 'file': 'base.yml'
  723. },
  724. 'environment': {'PROD': 'true'},
  725. },
  726. },
  727. }
  728. )
  729. tmpdir = pytest.ensuretemp('config_test')
  730. self.addCleanup(tmpdir.remove)
  731. tmpdir.join('base.yml').write("""
  732. version: '2.2'
  733. services:
  734. base:
  735. image: base
  736. web:
  737. extends: base
  738. """)
  739. details = config.ConfigDetails('.', [main_file])
  740. with tmpdir.as_cwd():
  741. service_dicts = config.load(details).services
  742. assert service_dicts[0] == {
  743. 'name': 'prodweb',
  744. 'image': 'base',
  745. 'environment': {'PROD': 'true'},
  746. }
  747. def test_load_with_multiple_files_and_invalid_override(self):
  748. base_file = config.ConfigFile(
  749. 'base.yaml',
  750. {'web': {'image': 'example/web'}})
  751. override_file = config.ConfigFile(
  752. 'override.yaml',
  753. {'bogus': 'thing'})
  754. details = config.ConfigDetails('.', [base_file, override_file])
  755. with pytest.raises(ConfigurationError) as exc:
  756. config.load(details)
  757. assert "service 'bogus' must be a mapping not a string." in exc.exconly()
  758. assert "In file 'override.yaml'" in exc.exconly()
  759. def test_load_sorts_in_dependency_order(self):
  760. config_details = build_config_details({
  761. 'web': {
  762. 'image': BUSYBOX_IMAGE_WITH_TAG,
  763. 'links': ['db'],
  764. },
  765. 'db': {
  766. 'image': BUSYBOX_IMAGE_WITH_TAG,
  767. 'volumes_from': ['volume:ro']
  768. },
  769. 'volume': {
  770. 'image': BUSYBOX_IMAGE_WITH_TAG,
  771. 'volumes': ['/tmp'],
  772. }
  773. })
  774. services = config.load(config_details).services
  775. assert services[0]['name'] == 'volume'
  776. assert services[1]['name'] == 'db'
  777. assert services[2]['name'] == 'web'
  778. def test_load_with_extensions(self):
  779. config_details = build_config_details({
  780. 'version': '2.3',
  781. 'x-data': {
  782. 'lambda': 3,
  783. 'excess': [True, {}]
  784. }
  785. })
  786. config_data = config.load(config_details)
  787. assert config_data.services == []
  788. def test_config_build_configuration(self):
  789. service = config.load(
  790. build_config_details(
  791. {'web': {
  792. 'build': '.',
  793. 'dockerfile': 'Dockerfile-alt'
  794. }},
  795. 'tests/fixtures/extends',
  796. 'filename.yml'
  797. )
  798. ).services
  799. assert 'context' in service[0]['build']
  800. assert service[0]['build']['dockerfile'] == 'Dockerfile-alt'
  801. def test_config_build_configuration_v2(self):
  802. # service.dockerfile is invalid in v2
  803. with pytest.raises(ConfigurationError):
  804. config.load(
  805. build_config_details(
  806. {
  807. 'version': '2',
  808. 'services': {
  809. 'web': {
  810. 'build': '.',
  811. 'dockerfile': 'Dockerfile-alt'
  812. }
  813. }
  814. },
  815. 'tests/fixtures/extends',
  816. 'filename.yml'
  817. )
  818. )
  819. service = config.load(
  820. build_config_details({
  821. 'version': '2',
  822. 'services': {
  823. 'web': {
  824. 'build': '.'
  825. }
  826. }
  827. }, 'tests/fixtures/extends', 'filename.yml')
  828. ).services[0]
  829. assert 'context' in service['build']
  830. service = config.load(
  831. build_config_details(
  832. {
  833. 'version': '2',
  834. 'services': {
  835. 'web': {
  836. 'build': {
  837. 'context': '.',
  838. 'dockerfile': 'Dockerfile-alt'
  839. }
  840. }
  841. }
  842. },
  843. 'tests/fixtures/extends',
  844. 'filename.yml'
  845. )
  846. ).services
  847. assert 'context' in service[0]['build']
  848. assert service[0]['build']['dockerfile'] == 'Dockerfile-alt'
  849. def test_load_with_buildargs(self):
  850. service = config.load(
  851. build_config_details(
  852. {
  853. 'version': '2',
  854. 'services': {
  855. 'web': {
  856. 'build': {
  857. 'context': '.',
  858. 'dockerfile': 'Dockerfile-alt',
  859. 'args': {
  860. 'opt1': 42,
  861. 'opt2': 'foobar'
  862. }
  863. }
  864. }
  865. }
  866. },
  867. 'tests/fixtures/extends',
  868. 'filename.yml'
  869. )
  870. ).services[0]
  871. assert 'args' in service['build']
  872. assert 'opt1' in service['build']['args']
  873. assert isinstance(service['build']['args']['opt1'], str)
  874. assert service['build']['args']['opt1'] == '42'
  875. assert service['build']['args']['opt2'] == 'foobar'
  876. def test_load_build_labels_dict(self):
  877. service = config.load(
  878. build_config_details(
  879. {
  880. 'version': str(V3_3),
  881. 'services': {
  882. 'web': {
  883. 'build': {
  884. 'context': '.',
  885. 'dockerfile': 'Dockerfile-alt',
  886. 'labels': {
  887. 'label1': 42,
  888. 'label2': 'foobar'
  889. }
  890. }
  891. }
  892. }
  893. },
  894. 'tests/fixtures/extends',
  895. 'filename.yml'
  896. )
  897. ).services[0]
  898. assert 'labels' in service['build']
  899. assert 'label1' in service['build']['labels']
  900. assert service['build']['labels']['label1'] == '42'
  901. assert service['build']['labels']['label2'] == 'foobar'
  902. def test_load_build_labels_list(self):
  903. base_file = config.ConfigFile(
  904. 'base.yml',
  905. {
  906. 'version': '2.3',
  907. 'services': {
  908. 'web': {
  909. 'build': {
  910. 'context': '.',
  911. 'labels': ['foo=bar', 'baz=true', 'foobar=1']
  912. },
  913. },
  914. },
  915. }
  916. )
  917. details = config.ConfigDetails('.', [base_file])
  918. service = config.load(details).services[0]
  919. assert service['build']['labels'] == {
  920. 'foo': 'bar', 'baz': 'true', 'foobar': '1'
  921. }
  922. def test_build_args_allow_empty_properties(self):
  923. service = config.load(
  924. build_config_details(
  925. {
  926. 'version': '2',
  927. 'services': {
  928. 'web': {
  929. 'build': {
  930. 'context': '.',
  931. 'dockerfile': 'Dockerfile-alt',
  932. 'args': {
  933. 'foo': None
  934. }
  935. }
  936. }
  937. }
  938. },
  939. 'tests/fixtures/extends',
  940. 'filename.yml'
  941. )
  942. ).services[0]
  943. assert 'args' in service['build']
  944. assert 'foo' in service['build']['args']
  945. assert service['build']['args']['foo'] == ''
  946. # If build argument is None then it will be converted to the empty
  947. # string. Make sure that int zero kept as it is, i.e. not converted to
  948. # the empty string
  949. def test_build_args_check_zero_preserved(self):
  950. service = config.load(
  951. build_config_details(
  952. {
  953. 'version': '2',
  954. 'services': {
  955. 'web': {
  956. 'build': {
  957. 'context': '.',
  958. 'dockerfile': 'Dockerfile-alt',
  959. 'args': {
  960. 'foo': 0
  961. }
  962. }
  963. }
  964. }
  965. },
  966. 'tests/fixtures/extends',
  967. 'filename.yml'
  968. )
  969. ).services[0]
  970. assert 'args' in service['build']
  971. assert 'foo' in service['build']['args']
  972. assert service['build']['args']['foo'] == '0'
  973. def test_load_with_multiple_files_mismatched_networks_format(self):
  974. base_file = config.ConfigFile(
  975. 'base.yaml',
  976. {
  977. 'version': '2',
  978. 'services': {
  979. 'web': {
  980. 'image': 'example/web',
  981. 'networks': {
  982. 'foobar': {'aliases': ['foo', 'bar']}
  983. }
  984. }
  985. },
  986. 'networks': {'foobar': {}, 'baz': {}}
  987. }
  988. )
  989. override_file = config.ConfigFile(
  990. 'override.yaml',
  991. {
  992. 'version': '2',
  993. 'services': {
  994. 'web': {
  995. 'networks': ['baz']
  996. }
  997. }
  998. }
  999. )
  1000. details = config.ConfigDetails('.', [base_file, override_file])
  1001. web_service = config.load(details).services[0]
  1002. assert web_service['networks'] == {
  1003. 'foobar': {'aliases': ['bar', 'foo']},
  1004. 'baz': {}
  1005. }
  1006. def test_load_with_multiple_files_mismatched_networks_format_inverse_order(self):
  1007. base_file = config.ConfigFile(
  1008. 'override.yaml',
  1009. {
  1010. 'version': '2',
  1011. 'services': {
  1012. 'web': {
  1013. 'networks': ['baz']
  1014. }
  1015. }
  1016. }
  1017. )
  1018. override_file = config.ConfigFile(
  1019. 'base.yaml',
  1020. {
  1021. 'version': '2',
  1022. 'services': {
  1023. 'web': {
  1024. 'image': 'example/web',
  1025. 'networks': {
  1026. 'foobar': {'aliases': ['foo', 'bar']}
  1027. }
  1028. }
  1029. },
  1030. 'networks': {'foobar': {}, 'baz': {}}
  1031. }
  1032. )
  1033. details = config.ConfigDetails('.', [base_file, override_file])
  1034. web_service = config.load(details).services[0]
  1035. assert web_service['networks'] == {
  1036. 'foobar': {'aliases': ['bar', 'foo']},
  1037. 'baz': {}
  1038. }
  1039. def test_load_with_multiple_files_v2(self):
  1040. base_file = config.ConfigFile(
  1041. 'base.yaml',
  1042. {
  1043. 'version': '2',
  1044. 'services': {
  1045. 'web': {
  1046. 'image': 'example/web',
  1047. 'depends_on': ['db'],
  1048. },
  1049. 'db': {
  1050. 'image': 'example/db',
  1051. }
  1052. },
  1053. })
  1054. override_file = config.ConfigFile(
  1055. 'override.yaml',
  1056. {
  1057. 'version': '2',
  1058. 'services': {
  1059. 'web': {
  1060. 'build': '/',
  1061. 'volumes': ['/home/user/project:/code'],
  1062. 'depends_on': ['other'],
  1063. },
  1064. 'other': {
  1065. 'image': 'example/other',
  1066. }
  1067. }
  1068. })
  1069. details = config.ConfigDetails('.', [base_file, override_file])
  1070. service_dicts = config.load(details).services
  1071. expected = [
  1072. {
  1073. 'name': 'web',
  1074. 'build': {'context': os.path.abspath('/')},
  1075. 'image': 'example/web',
  1076. 'volumes': [VolumeSpec.parse('/home/user/project:/code')],
  1077. 'depends_on': {
  1078. 'db': {'condition': 'service_started'},
  1079. 'other': {'condition': 'service_started'},
  1080. },
  1081. },
  1082. {
  1083. 'name': 'db',
  1084. 'image': 'example/db',
  1085. },
  1086. {
  1087. 'name': 'other',
  1088. 'image': 'example/other',
  1089. },
  1090. ]
  1091. assert service_sort(service_dicts) == service_sort(expected)
  1092. @mock.patch.dict(os.environ)
  1093. def test_load_with_multiple_files_v3_2(self):
  1094. os.environ['COMPOSE_CONVERT_WINDOWS_PATHS'] = 'true'
  1095. base_file = config.ConfigFile(
  1096. 'base.yaml',
  1097. {
  1098. 'version': '3.2',
  1099. 'services': {
  1100. 'web': {
  1101. 'image': 'example/web',
  1102. 'volumes': [
  1103. {'source': '/a', 'target': '/b', 'type': 'bind'},
  1104. {'source': 'vol', 'target': '/x', 'type': 'volume', 'read_only': True}
  1105. ],
  1106. 'stop_grace_period': '30s',
  1107. }
  1108. },
  1109. 'volumes': {'vol': {}}
  1110. }
  1111. )
  1112. override_file = config.ConfigFile(
  1113. 'override.yaml',
  1114. {
  1115. 'version': '3.2',
  1116. 'services': {
  1117. 'web': {
  1118. 'volumes': ['/c:/b', '/anonymous']
  1119. }
  1120. }
  1121. }
  1122. )
  1123. details = config.ConfigDetails('.', [base_file, override_file])
  1124. service_dicts = config.load(details).services
  1125. svc_volumes = map(lambda v: v.repr(), service_dicts[0]['volumes'])
  1126. for vol in svc_volumes:
  1127. assert vol in [
  1128. '/anonymous',
  1129. '/c:/b:rw',
  1130. {'source': 'vol', 'target': '/x', 'type': 'volume', 'read_only': True}
  1131. ]
  1132. assert service_dicts[0]['stop_grace_period'] == '30s'
  1133. @mock.patch.dict(os.environ)
  1134. def test_volume_mode_override(self):
  1135. os.environ['COMPOSE_CONVERT_WINDOWS_PATHS'] = 'true'
  1136. base_file = config.ConfigFile(
  1137. 'base.yaml',
  1138. {
  1139. 'version': '2.3',
  1140. 'services': {
  1141. 'web': {
  1142. 'image': 'example/web',
  1143. 'volumes': ['/c:/b:rw']
  1144. }
  1145. },
  1146. }
  1147. )
  1148. override_file = config.ConfigFile(
  1149. 'override.yaml',
  1150. {
  1151. 'version': '2.3',
  1152. 'services': {
  1153. 'web': {
  1154. 'volumes': ['/c:/b:ro']
  1155. }
  1156. }
  1157. }
  1158. )
  1159. details = config.ConfigDetails('.', [base_file, override_file])
  1160. service_dicts = config.load(details).services
  1161. svc_volumes = list(map(lambda v: v.repr(), service_dicts[0]['volumes']))
  1162. assert svc_volumes == ['/c:/b:ro']
  1163. def test_undeclared_volume_v2(self):
  1164. base_file = config.ConfigFile(
  1165. 'base.yaml',
  1166. {
  1167. 'version': '2',
  1168. 'services': {
  1169. 'web': {
  1170. 'image': BUSYBOX_IMAGE_WITH_TAG,
  1171. 'volumes': ['data0028:/data:ro'],
  1172. },
  1173. },
  1174. }
  1175. )
  1176. details = config.ConfigDetails('.', [base_file])
  1177. with pytest.raises(ConfigurationError):
  1178. config.load(details)
  1179. base_file = config.ConfigFile(
  1180. 'base.yaml',
  1181. {
  1182. 'version': '2',
  1183. 'services': {
  1184. 'web': {
  1185. 'image': BUSYBOX_IMAGE_WITH_TAG,
  1186. 'volumes': ['./data0028:/data:ro'],
  1187. },
  1188. },
  1189. }
  1190. )
  1191. details = config.ConfigDetails('.', [base_file])
  1192. config_data = config.load(details)
  1193. volume = config_data.services[0].get('volumes')[0]
  1194. assert not volume.is_named_volume
  1195. def test_undeclared_volume_v1(self):
  1196. base_file = config.ConfigFile(
  1197. 'base.yaml',
  1198. {
  1199. 'web': {
  1200. 'image': BUSYBOX_IMAGE_WITH_TAG,
  1201. 'volumes': ['data0028:/data:ro'],
  1202. },
  1203. }
  1204. )
  1205. details = config.ConfigDetails('.', [base_file])
  1206. config_data = config.load(details)
  1207. volume = config_data.services[0].get('volumes')[0]
  1208. assert volume.external == 'data0028'
  1209. assert volume.is_named_volume
  1210. def test_volumes_long_syntax(self):
  1211. base_file = config.ConfigFile(
  1212. 'base.yaml', {
  1213. 'version': '2.3',
  1214. 'services': {
  1215. 'web': {
  1216. 'image': BUSYBOX_IMAGE_WITH_TAG,
  1217. 'volumes': [
  1218. {
  1219. 'target': '/anonymous', 'type': 'volume'
  1220. }, {
  1221. 'source': '/abc', 'target': '/xyz', 'type': 'bind'
  1222. }, {
  1223. 'source': '\\\\.\\pipe\\abcd', 'target': '/named_pipe', 'type': 'npipe'
  1224. }, {
  1225. 'type': 'tmpfs', 'target': '/tmpfs'
  1226. }
  1227. ]
  1228. },
  1229. },
  1230. },
  1231. )
  1232. details = config.ConfigDetails('.', [base_file])
  1233. config_data = config.load(details)
  1234. volumes = config_data.services[0].get('volumes')
  1235. anon_volume = [v for v in volumes if v.target == '/anonymous'][0]
  1236. tmpfs_mount = [v for v in volumes if v.type == 'tmpfs'][0]
  1237. host_mount = [v for v in volumes if v.type == 'bind'][0]
  1238. npipe_mount = [v for v in volumes if v.type == 'npipe'][0]
  1239. assert anon_volume.type == 'volume'
  1240. assert not anon_volume.is_named_volume
  1241. assert tmpfs_mount.target == '/tmpfs'
  1242. assert not tmpfs_mount.is_named_volume
  1243. assert host_mount.source == '/abc'
  1244. assert host_mount.target == '/xyz'
  1245. assert not host_mount.is_named_volume
  1246. assert npipe_mount.source == '\\\\.\\pipe\\abcd'
  1247. assert npipe_mount.target == '/named_pipe'
  1248. assert not npipe_mount.is_named_volume
  1249. def test_load_bind_mount_relative_path(self):
  1250. expected_source = 'C:\\tmp\\web' if IS_WINDOWS_PLATFORM else '/tmp/web'
  1251. base_file = config.ConfigFile(
  1252. 'base.yaml', {
  1253. 'version': '3.4',
  1254. 'services': {
  1255. 'web': {
  1256. 'image': BUSYBOX_IMAGE_WITH_TAG,
  1257. 'volumes': [
  1258. {'type': 'bind', 'source': './web', 'target': '/web'},
  1259. ],
  1260. },
  1261. },
  1262. },
  1263. )
  1264. details = config.ConfigDetails('/tmp', [base_file])
  1265. config_data = config.load(details)
  1266. mount = config_data.services[0].get('volumes')[0]
  1267. assert mount.target == '/web'
  1268. assert mount.type == 'bind'
  1269. assert mount.source == expected_source
  1270. def test_load_bind_mount_relative_path_with_tilde(self):
  1271. base_file = config.ConfigFile(
  1272. 'base.yaml', {
  1273. 'version': '3.4',
  1274. 'services': {
  1275. 'web': {
  1276. 'image': BUSYBOX_IMAGE_WITH_TAG,
  1277. 'volumes': [
  1278. {'type': 'bind', 'source': '~/web', 'target': '/web'},
  1279. ],
  1280. },
  1281. },
  1282. },
  1283. )
  1284. details = config.ConfigDetails('.', [base_file])
  1285. config_data = config.load(details)
  1286. mount = config_data.services[0].get('volumes')[0]
  1287. assert mount.target == '/web'
  1288. assert mount.type == 'bind'
  1289. assert (
  1290. not mount.source.startswith('~') and mount.source.endswith(
  1291. '{}web'.format(os.path.sep)
  1292. )
  1293. )
  1294. def test_config_invalid_ipam_config(self):
  1295. with pytest.raises(ConfigurationError) as excinfo:
  1296. config.load(
  1297. build_config_details(
  1298. {
  1299. 'version': str(V2_1),
  1300. 'networks': {
  1301. 'foo': {
  1302. 'driver': 'default',
  1303. 'ipam': {
  1304. 'driver': 'default',
  1305. 'config': ['172.18.0.0/16'],
  1306. }
  1307. }
  1308. }
  1309. },
  1310. filename='filename.yml',
  1311. )
  1312. )
  1313. assert ('networks.foo.ipam.config contains an invalid type,'
  1314. ' it should be an object') in excinfo.exconly()
  1315. def test_config_valid_ipam_config(self):
  1316. ipam_config = {
  1317. 'subnet': '172.28.0.0/16',
  1318. 'ip_range': '172.28.5.0/24',
  1319. 'gateway': '172.28.5.254',
  1320. 'aux_addresses': {
  1321. 'host1': '172.28.1.5',
  1322. 'host2': '172.28.1.6',
  1323. 'host3': '172.28.1.7',
  1324. },
  1325. }
  1326. networks = config.load(
  1327. build_config_details(
  1328. {
  1329. 'version': str(V2_1),
  1330. 'networks': {
  1331. 'foo': {
  1332. 'driver': 'default',
  1333. 'ipam': {
  1334. 'driver': 'default',
  1335. 'config': [ipam_config],
  1336. }
  1337. }
  1338. }
  1339. },
  1340. filename='filename.yml',
  1341. )
  1342. ).networks
  1343. assert 'foo' in networks
  1344. assert networks['foo']['ipam']['config'] == [ipam_config]
  1345. def test_config_valid_service_names(self):
  1346. for valid_name in ['_', '-', '.__.', '_what-up.', 'what_.up----', 'whatup']:
  1347. services = config.load(
  1348. build_config_details(
  1349. {valid_name: {'image': 'busybox'}},
  1350. 'tests/fixtures/extends',
  1351. 'common.yml')).services
  1352. assert services[0]['name'] == valid_name
  1353. def test_config_hint(self):
  1354. with pytest.raises(ConfigurationError) as excinfo:
  1355. config.load(
  1356. build_config_details(
  1357. {
  1358. 'foo': {'image': 'busybox', 'privilige': 'something'},
  1359. },
  1360. 'tests/fixtures/extends',
  1361. 'filename.yml'
  1362. )
  1363. )
  1364. assert "(did you mean 'privileged'?)" in excinfo.exconly()
  1365. def test_load_errors_on_uppercase_with_no_image(self):
  1366. with pytest.raises(ConfigurationError) as exc:
  1367. config.load(build_config_details({
  1368. 'Foo': {'build': '.'},
  1369. }, 'tests/fixtures/build-ctx'))
  1370. assert "Service 'Foo' contains uppercase characters" in exc.exconly()
  1371. def test_invalid_config_v1(self):
  1372. with pytest.raises(ConfigurationError) as excinfo:
  1373. config.load(
  1374. build_config_details(
  1375. {
  1376. 'foo': {'image': 1},
  1377. },
  1378. 'tests/fixtures/extends',
  1379. 'filename.yml'
  1380. )
  1381. )
  1382. assert "foo.image contains an invalid type, it should be a string" \
  1383. in excinfo.exconly()
  1384. def test_invalid_config_v2(self):
  1385. with pytest.raises(ConfigurationError) as excinfo:
  1386. config.load(
  1387. build_config_details(
  1388. {
  1389. 'version': '2',
  1390. 'services': {
  1391. 'foo': {'image': 1},
  1392. },
  1393. },
  1394. 'tests/fixtures/extends',
  1395. 'filename.yml'
  1396. )
  1397. )
  1398. assert "services.foo.image contains an invalid type, it should be a string" \
  1399. in excinfo.exconly()
  1400. def test_invalid_config_build_and_image_specified_v1(self):
  1401. with pytest.raises(ConfigurationError) as excinfo:
  1402. config.load(
  1403. build_config_details(
  1404. {
  1405. 'foo': {'image': 'busybox', 'build': '.'},
  1406. },
  1407. 'tests/fixtures/extends',
  1408. 'filename.yml'
  1409. )
  1410. )
  1411. assert "foo has both an image and build path specified." in excinfo.exconly()
  1412. def test_invalid_config_type_should_be_an_array(self):
  1413. with pytest.raises(ConfigurationError) as excinfo:
  1414. config.load(
  1415. build_config_details(
  1416. {
  1417. 'foo': {'image': 'busybox', 'links': 'an_link'},
  1418. },
  1419. 'tests/fixtures/extends',
  1420. 'filename.yml'
  1421. )
  1422. )
  1423. assert "foo.links contains an invalid type, it should be an array" \
  1424. in excinfo.exconly()
  1425. def test_invalid_config_not_a_dictionary(self):
  1426. with pytest.raises(ConfigurationError) as excinfo:
  1427. config.load(
  1428. build_config_details(
  1429. ['foo', 'lol'],
  1430. 'tests/fixtures/extends',
  1431. 'filename.yml'
  1432. )
  1433. )
  1434. assert "Top level object in 'filename.yml' needs to be an object" \
  1435. in excinfo.exconly()
  1436. def test_invalid_config_not_unique_items(self):
  1437. with pytest.raises(ConfigurationError) as excinfo:
  1438. config.load(
  1439. build_config_details(
  1440. {
  1441. 'web': {'build': '.', 'devices': ['/dev/foo:/dev/foo', '/dev/foo:/dev/foo']}
  1442. },
  1443. 'tests/fixtures/extends',
  1444. 'filename.yml'
  1445. )
  1446. )
  1447. assert "has non-unique elements" in excinfo.exconly()
  1448. def test_invalid_list_of_strings_format(self):
  1449. with pytest.raises(ConfigurationError) as excinfo:
  1450. config.load(
  1451. build_config_details(
  1452. {
  1453. 'web': {'build': '.', 'command': [1]}
  1454. },
  1455. 'tests/fixtures/extends',
  1456. 'filename.yml'
  1457. )
  1458. )
  1459. assert "web.command contains 1, which is an invalid type, it should be a string" \
  1460. in excinfo.exconly()
  1461. def test_load_config_dockerfile_without_build_raises_error_v1(self):
  1462. with pytest.raises(ConfigurationError) as exc:
  1463. config.load(build_config_details({
  1464. 'web': {
  1465. 'image': 'busybox',
  1466. 'dockerfile': 'Dockerfile.alt'
  1467. }
  1468. }))
  1469. assert "web has both an image and alternate Dockerfile." in exc.exconly()
  1470. def test_config_extra_hosts_string_raises_validation_error(self):
  1471. with pytest.raises(ConfigurationError) as excinfo:
  1472. config.load(
  1473. build_config_details(
  1474. {'web': {
  1475. 'image': 'busybox',
  1476. 'extra_hosts': 'somehost:162.242.195.82'
  1477. }},
  1478. 'working_dir',
  1479. 'filename.yml'
  1480. )
  1481. )
  1482. assert "web.extra_hosts contains an invalid type" \
  1483. in excinfo.exconly()
  1484. def test_config_extra_hosts_list_of_dicts_validation_error(self):
  1485. with pytest.raises(ConfigurationError) as excinfo:
  1486. config.load(
  1487. build_config_details(
  1488. {'web': {
  1489. 'image': 'busybox',
  1490. 'extra_hosts': [
  1491. {'somehost': '162.242.195.82'},
  1492. {'otherhost': '50.31.209.229'}
  1493. ]
  1494. }},
  1495. 'working_dir',
  1496. 'filename.yml'
  1497. )
  1498. )
  1499. assert "web.extra_hosts contains {\"somehost\": \"162.242.195.82\"}, " \
  1500. "which is an invalid type, it should be a string" \
  1501. in excinfo.exconly()
  1502. def test_config_ulimits_invalid_keys_validation_error(self):
  1503. with pytest.raises(ConfigurationError) as exc:
  1504. config.load(build_config_details(
  1505. {
  1506. 'web': {
  1507. 'image': 'busybox',
  1508. 'ulimits': {
  1509. 'nofile': {
  1510. "not_soft_or_hard": 100,
  1511. "soft": 10000,
  1512. "hard": 20000,
  1513. }
  1514. }
  1515. }
  1516. },
  1517. 'working_dir',
  1518. 'filename.yml'))
  1519. assert "web.ulimits.nofile contains unsupported option: 'not_soft_or_hard'" \
  1520. in exc.exconly()
  1521. def test_config_ulimits_required_keys_validation_error(self):
  1522. with pytest.raises(ConfigurationError) as exc:
  1523. config.load(build_config_details(
  1524. {
  1525. 'web': {
  1526. 'image': 'busybox',
  1527. 'ulimits': {'nofile': {"soft": 10000}}
  1528. }
  1529. },
  1530. 'working_dir',
  1531. 'filename.yml'))
  1532. assert "web.ulimits.nofile" in exc.exconly()
  1533. assert "'hard' is a required property" in exc.exconly()
  1534. def test_config_ulimits_soft_greater_than_hard_error(self):
  1535. expected = "'soft' value can not be greater than 'hard' value"
  1536. with pytest.raises(ConfigurationError) as exc:
  1537. config.load(build_config_details(
  1538. {
  1539. 'web': {
  1540. 'image': 'busybox',
  1541. 'ulimits': {
  1542. 'nofile': {"soft": 10000, "hard": 1000}
  1543. }
  1544. }
  1545. },
  1546. 'working_dir',
  1547. 'filename.yml'))
  1548. assert expected in exc.exconly()
  1549. def test_valid_config_which_allows_two_type_definitions(self):
  1550. expose_values = [["8000"], [8000]]
  1551. for expose in expose_values:
  1552. service = config.load(
  1553. build_config_details(
  1554. {'web': {
  1555. 'image': 'busybox',
  1556. 'expose': expose
  1557. }},
  1558. 'working_dir',
  1559. 'filename.yml'
  1560. )
  1561. ).services
  1562. assert service[0]['expose'] == expose
  1563. def test_valid_config_oneof_string_or_list(self):
  1564. entrypoint_values = [["sh"], "sh"]
  1565. for entrypoint in entrypoint_values:
  1566. service = config.load(
  1567. build_config_details(
  1568. {'web': {
  1569. 'image': 'busybox',
  1570. 'entrypoint': entrypoint
  1571. }},
  1572. 'working_dir',
  1573. 'filename.yml'
  1574. )
  1575. ).services
  1576. assert service[0]['entrypoint'] == entrypoint
  1577. def test_logs_warning_for_boolean_in_environment(self):
  1578. config_details = build_config_details({
  1579. 'web': {
  1580. 'image': 'busybox',
  1581. 'environment': {'SHOW_STUFF': True}
  1582. }
  1583. })
  1584. with pytest.raises(ConfigurationError) as exc:
  1585. config.load(config_details)
  1586. assert "contains true, which is an invalid type" in exc.exconly()
  1587. def test_config_valid_environment_dict_key_contains_dashes(self):
  1588. services = config.load(
  1589. build_config_details(
  1590. {'web': {
  1591. 'image': 'busybox',
  1592. 'environment': {'SPRING_JPA_HIBERNATE_DDL-AUTO': 'none'}
  1593. }},
  1594. 'working_dir',
  1595. 'filename.yml'
  1596. )
  1597. ).services
  1598. assert services[0]['environment']['SPRING_JPA_HIBERNATE_DDL-AUTO'] == 'none'
  1599. def test_load_yaml_with_yaml_error(self):
  1600. tmpdir = py.test.ensuretemp('invalid_yaml_test')
  1601. self.addCleanup(tmpdir.remove)
  1602. invalid_yaml_file = tmpdir.join('docker-compose.yml')
  1603. invalid_yaml_file.write("""
  1604. web:
  1605. this is bogus: ok: what
  1606. """)
  1607. with pytest.raises(ConfigurationError) as exc:
  1608. config.load_yaml(str(invalid_yaml_file))
  1609. assert 'line 3, column 32' in exc.exconly()
  1610. def test_load_yaml_with_bom(self):
  1611. tmpdir = py.test.ensuretemp('bom_yaml')
  1612. self.addCleanup(tmpdir.remove)
  1613. bom_yaml = tmpdir.join('docker-compose.yml')
  1614. with codecs.open(str(bom_yaml), 'w', encoding='utf-8') as f:
  1615. f.write('''\ufeff
  1616. version: '2.3'
  1617. volumes:
  1618. park_bom:
  1619. ''')
  1620. assert config.load_yaml(str(bom_yaml)) == {
  1621. 'version': '2.3',
  1622. 'volumes': {'park_bom': None}
  1623. }
  1624. def test_validate_extra_hosts_invalid(self):
  1625. with pytest.raises(ConfigurationError) as exc:
  1626. config.load(build_config_details({
  1627. 'web': {
  1628. 'image': 'alpine',
  1629. 'extra_hosts': "www.example.com: 192.168.0.17",
  1630. }
  1631. }))
  1632. assert "web.extra_hosts contains an invalid type" in exc.exconly()
  1633. def test_validate_extra_hosts_invalid_list(self):
  1634. with pytest.raises(ConfigurationError) as exc:
  1635. config.load(build_config_details({
  1636. 'web': {
  1637. 'image': 'alpine',
  1638. 'extra_hosts': [
  1639. {'www.example.com': '192.168.0.17'},
  1640. {'api.example.com': '192.168.0.18'}
  1641. ],
  1642. }
  1643. }))
  1644. assert "which is an invalid type" in exc.exconly()
  1645. def test_normalize_dns_options(self):
  1646. actual = config.load(build_config_details({
  1647. 'web': {
  1648. 'image': 'alpine',
  1649. 'dns': '8.8.8.8',
  1650. 'dns_search': 'domain.local',
  1651. }
  1652. }))
  1653. assert actual.services == [
  1654. {
  1655. 'name': 'web',
  1656. 'image': 'alpine',
  1657. 'dns': ['8.8.8.8'],
  1658. 'dns_search': ['domain.local'],
  1659. }
  1660. ]
  1661. def test_tmpfs_option(self):
  1662. actual = config.load(build_config_details({
  1663. 'version': '2',
  1664. 'services': {
  1665. 'web': {
  1666. 'image': 'alpine',
  1667. 'tmpfs': '/run',
  1668. }
  1669. }
  1670. }))
  1671. assert actual.services == [
  1672. {
  1673. 'name': 'web',
  1674. 'image': 'alpine',
  1675. 'tmpfs': ['/run'],
  1676. }
  1677. ]
  1678. def test_oom_score_adj_option(self):
  1679. actual = config.load(build_config_details({
  1680. 'version': '2',
  1681. 'services': {
  1682. 'web': {
  1683. 'image': 'alpine',
  1684. 'oom_score_adj': 500
  1685. }
  1686. }
  1687. }))
  1688. assert actual.services == [
  1689. {
  1690. 'name': 'web',
  1691. 'image': 'alpine',
  1692. 'oom_score_adj': 500
  1693. }
  1694. ]
  1695. def test_swappiness_option(self):
  1696. actual = config.load(build_config_details({
  1697. 'version': '2',
  1698. 'services': {
  1699. 'web': {
  1700. 'image': 'alpine',
  1701. 'mem_swappiness': 10,
  1702. }
  1703. }
  1704. }))
  1705. assert actual.services == [
  1706. {
  1707. 'name': 'web',
  1708. 'image': 'alpine',
  1709. 'mem_swappiness': 10,
  1710. }
  1711. ]
  1712. def test_group_add_option(self):
  1713. actual = config.load(build_config_details({
  1714. 'version': '2',
  1715. 'services': {
  1716. 'web': {
  1717. 'image': 'alpine',
  1718. 'group_add': ["docker", 777]
  1719. }
  1720. }
  1721. }))
  1722. assert actual.services == [
  1723. {
  1724. 'name': 'web',
  1725. 'image': 'alpine',
  1726. 'group_add': ["docker", 777]
  1727. }
  1728. ]
  1729. def test_dns_opt_option(self):
  1730. actual = config.load(build_config_details({
  1731. 'version': '2',
  1732. 'services': {
  1733. 'web': {
  1734. 'image': 'alpine',
  1735. 'dns_opt': ["use-vc", "no-tld-query"]
  1736. }
  1737. }
  1738. }))
  1739. assert actual.services == [
  1740. {
  1741. 'name': 'web',
  1742. 'image': 'alpine',
  1743. 'dns_opt': ["use-vc", "no-tld-query"]
  1744. }
  1745. ]
  1746. def test_isolation_option(self):
  1747. actual = config.load(build_config_details({
  1748. 'version': str(V2_1),
  1749. 'services': {
  1750. 'web': {
  1751. 'image': 'win10',
  1752. 'isolation': 'hyperv'
  1753. }
  1754. }
  1755. }))
  1756. assert actual.services == [
  1757. {
  1758. 'name': 'web',
  1759. 'image': 'win10',
  1760. 'isolation': 'hyperv',
  1761. }
  1762. ]
  1763. def test_runtime_option(self):
  1764. actual = config.load(build_config_details({
  1765. 'version': str(V2_3),
  1766. 'services': {
  1767. 'web': {
  1768. 'image': 'nvidia/cuda',
  1769. 'runtime': 'nvidia'
  1770. }
  1771. }
  1772. }))
  1773. assert actual.services == [
  1774. {
  1775. 'name': 'web',
  1776. 'image': 'nvidia/cuda',
  1777. 'runtime': 'nvidia',
  1778. }
  1779. ]
  1780. def test_merge_service_dicts_from_files_with_extends_in_base(self):
  1781. base = {
  1782. 'volumes': ['.:/app'],
  1783. 'extends': {'service': 'app'}
  1784. }
  1785. override = {
  1786. 'image': 'alpine:edge',
  1787. }
  1788. actual = config.merge_service_dicts_from_files(
  1789. base,
  1790. override,
  1791. DEFAULT_VERSION)
  1792. assert actual == {
  1793. 'image': 'alpine:edge',
  1794. 'volumes': ['.:/app'],
  1795. 'extends': {'service': 'app'}
  1796. }
  1797. def test_merge_service_dicts_from_files_with_extends_in_override(self):
  1798. base = {
  1799. 'volumes': ['.:/app'],
  1800. 'extends': {'service': 'app'}
  1801. }
  1802. override = {
  1803. 'image': 'alpine:edge',
  1804. 'extends': {'service': 'foo'}
  1805. }
  1806. actual = config.merge_service_dicts_from_files(
  1807. base,
  1808. override,
  1809. DEFAULT_VERSION)
  1810. assert actual == {
  1811. 'image': 'alpine:edge',
  1812. 'volumes': ['.:/app'],
  1813. 'extends': {'service': 'foo'}
  1814. }
  1815. def test_merge_service_dicts_heterogeneous(self):
  1816. base = {
  1817. 'volumes': ['.:/app'],
  1818. 'ports': ['5432']
  1819. }
  1820. override = {
  1821. 'image': 'alpine:edge',
  1822. 'ports': [5432]
  1823. }
  1824. actual = config.merge_service_dicts_from_files(
  1825. base,
  1826. override,
  1827. DEFAULT_VERSION)
  1828. assert actual == {
  1829. 'image': 'alpine:edge',
  1830. 'volumes': ['.:/app'],
  1831. 'ports': types.ServicePort.parse('5432')
  1832. }
  1833. def test_merge_service_dicts_heterogeneous_2(self):
  1834. base = {
  1835. 'volumes': ['.:/app'],
  1836. 'ports': [5432]
  1837. }
  1838. override = {
  1839. 'image': 'alpine:edge',
  1840. 'ports': ['5432']
  1841. }
  1842. actual = config.merge_service_dicts_from_files(
  1843. base,
  1844. override,
  1845. DEFAULT_VERSION)
  1846. assert actual == {
  1847. 'image': 'alpine:edge',
  1848. 'volumes': ['.:/app'],
  1849. 'ports': types.ServicePort.parse('5432')
  1850. }
  1851. def test_merge_service_dicts_ports_sorting(self):
  1852. base = {
  1853. 'ports': [5432]
  1854. }
  1855. override = {
  1856. 'image': 'alpine:edge',
  1857. 'ports': ['5432/udp']
  1858. }
  1859. actual = config.merge_service_dicts_from_files(
  1860. base,
  1861. override,
  1862. DEFAULT_VERSION)
  1863. assert len(actual['ports']) == 2
  1864. assert types.ServicePort.parse('5432')[0] in actual['ports']
  1865. assert types.ServicePort.parse('5432/udp')[0] in actual['ports']
  1866. def test_merge_service_dicts_heterogeneous_volumes(self):
  1867. base = {
  1868. 'volumes': ['/a:/b', '/x:/z'],
  1869. }
  1870. override = {
  1871. 'image': 'alpine:edge',
  1872. 'volumes': [
  1873. {'source': '/e', 'target': '/b', 'type': 'bind'},
  1874. {'source': '/c', 'target': '/d', 'type': 'bind'}
  1875. ]
  1876. }
  1877. actual = config.merge_service_dicts_from_files(
  1878. base, override, V3_2
  1879. )
  1880. assert actual['volumes'] == [
  1881. {'source': '/e', 'target': '/b', 'type': 'bind'},
  1882. {'source': '/c', 'target': '/d', 'type': 'bind'},
  1883. '/x:/z'
  1884. ]
  1885. def test_merge_logging_v1(self):
  1886. base = {
  1887. 'image': 'alpine:edge',
  1888. 'log_driver': 'something',
  1889. 'log_opt': {'foo': 'three'},
  1890. }
  1891. override = {
  1892. 'image': 'alpine:edge',
  1893. 'command': 'true',
  1894. }
  1895. actual = config.merge_service_dicts(base, override, V1)
  1896. assert actual == {
  1897. 'image': 'alpine:edge',
  1898. 'log_driver': 'something',
  1899. 'log_opt': {'foo': 'three'},
  1900. 'command': 'true',
  1901. }
  1902. def test_merge_logging_v2(self):
  1903. base = {
  1904. 'image': 'alpine:edge',
  1905. 'logging': {
  1906. 'driver': 'json-file',
  1907. 'options': {
  1908. 'frequency': '2000',
  1909. 'timeout': '23'
  1910. }
  1911. }
  1912. }
  1913. override = {
  1914. 'logging': {
  1915. 'options': {
  1916. 'timeout': '360',
  1917. 'pretty-print': 'on'
  1918. }
  1919. }
  1920. }
  1921. actual = config.merge_service_dicts(base, override, V2_0)
  1922. assert actual == {
  1923. 'image': 'alpine:edge',
  1924. 'logging': {
  1925. 'driver': 'json-file',
  1926. 'options': {
  1927. 'frequency': '2000',
  1928. 'timeout': '360',
  1929. 'pretty-print': 'on'
  1930. }
  1931. }
  1932. }
  1933. def test_merge_logging_v2_override_driver(self):
  1934. base = {
  1935. 'image': 'alpine:edge',
  1936. 'logging': {
  1937. 'driver': 'json-file',
  1938. 'options': {
  1939. 'frequency': '2000',
  1940. 'timeout': '23'
  1941. }
  1942. }
  1943. }
  1944. override = {
  1945. 'logging': {
  1946. 'driver': 'syslog',
  1947. 'options': {
  1948. 'timeout': '360',
  1949. 'pretty-print': 'on'
  1950. }
  1951. }
  1952. }
  1953. actual = config.merge_service_dicts(base, override, V2_0)
  1954. assert actual == {
  1955. 'image': 'alpine:edge',
  1956. 'logging': {
  1957. 'driver': 'syslog',
  1958. 'options': {
  1959. 'timeout': '360',
  1960. 'pretty-print': 'on'
  1961. }
  1962. }
  1963. }
  1964. def test_merge_logging_v2_no_base_driver(self):
  1965. base = {
  1966. 'image': 'alpine:edge',
  1967. 'logging': {
  1968. 'options': {
  1969. 'frequency': '2000',
  1970. 'timeout': '23'
  1971. }
  1972. }
  1973. }
  1974. override = {
  1975. 'logging': {
  1976. 'driver': 'json-file',
  1977. 'options': {
  1978. 'timeout': '360',
  1979. 'pretty-print': 'on'
  1980. }
  1981. }
  1982. }
  1983. actual = config.merge_service_dicts(base, override, V2_0)
  1984. assert actual == {
  1985. 'image': 'alpine:edge',
  1986. 'logging': {
  1987. 'driver': 'json-file',
  1988. 'options': {
  1989. 'frequency': '2000',
  1990. 'timeout': '360',
  1991. 'pretty-print': 'on'
  1992. }
  1993. }
  1994. }
  1995. def test_merge_logging_v2_no_drivers(self):
  1996. base = {
  1997. 'image': 'alpine:edge',
  1998. 'logging': {
  1999. 'options': {
  2000. 'frequency': '2000',
  2001. 'timeout': '23'
  2002. }
  2003. }
  2004. }
  2005. override = {
  2006. 'logging': {
  2007. 'options': {
  2008. 'timeout': '360',
  2009. 'pretty-print': 'on'
  2010. }
  2011. }
  2012. }
  2013. actual = config.merge_service_dicts(base, override, V2_0)
  2014. assert actual == {
  2015. 'image': 'alpine:edge',
  2016. 'logging': {
  2017. 'options': {
  2018. 'frequency': '2000',
  2019. 'timeout': '360',
  2020. 'pretty-print': 'on'
  2021. }
  2022. }
  2023. }
  2024. def test_merge_logging_v2_no_override_options(self):
  2025. base = {
  2026. 'image': 'alpine:edge',
  2027. 'logging': {
  2028. 'driver': 'json-file',
  2029. 'options': {
  2030. 'frequency': '2000',
  2031. 'timeout': '23'
  2032. }
  2033. }
  2034. }
  2035. override = {
  2036. 'logging': {
  2037. 'driver': 'syslog'
  2038. }
  2039. }
  2040. actual = config.merge_service_dicts(base, override, V2_0)
  2041. assert actual == {
  2042. 'image': 'alpine:edge',
  2043. 'logging': {
  2044. 'driver': 'syslog',
  2045. }
  2046. }
  2047. def test_merge_logging_v2_no_base(self):
  2048. base = {
  2049. 'image': 'alpine:edge'
  2050. }
  2051. override = {
  2052. 'logging': {
  2053. 'driver': 'json-file',
  2054. 'options': {
  2055. 'frequency': '2000'
  2056. }
  2057. }
  2058. }
  2059. actual = config.merge_service_dicts(base, override, V2_0)
  2060. assert actual == {
  2061. 'image': 'alpine:edge',
  2062. 'logging': {
  2063. 'driver': 'json-file',
  2064. 'options': {
  2065. 'frequency': '2000'
  2066. }
  2067. }
  2068. }
  2069. def test_merge_logging_v2_no_override(self):
  2070. base = {
  2071. 'image': 'alpine:edge',
  2072. 'logging': {
  2073. 'driver': 'syslog',
  2074. 'options': {
  2075. 'frequency': '2000'
  2076. }
  2077. }
  2078. }
  2079. override = {}
  2080. actual = config.merge_service_dicts(base, override, V2_0)
  2081. assert actual == {
  2082. 'image': 'alpine:edge',
  2083. 'logging': {
  2084. 'driver': 'syslog',
  2085. 'options': {
  2086. 'frequency': '2000'
  2087. }
  2088. }
  2089. }
  2090. def test_merge_mixed_ports(self):
  2091. base = {
  2092. 'image': BUSYBOX_IMAGE_WITH_TAG,
  2093. 'command': 'top',
  2094. 'ports': [
  2095. {
  2096. 'target': '1245',
  2097. 'published': '1245',
  2098. 'protocol': 'udp',
  2099. }
  2100. ]
  2101. }
  2102. override = {
  2103. 'ports': ['1245:1245/udp']
  2104. }
  2105. actual = config.merge_service_dicts(base, override, V3_1)
  2106. assert actual == {
  2107. 'image': BUSYBOX_IMAGE_WITH_TAG,
  2108. 'command': 'top',
  2109. 'ports': [types.ServicePort('1245', '1245', 'udp', None, None)]
  2110. }
  2111. def test_merge_depends_on_no_override(self):
  2112. base = {
  2113. 'image': 'busybox',
  2114. 'depends_on': {
  2115. 'app1': {'condition': 'service_started'},
  2116. 'app2': {'condition': 'service_healthy'}
  2117. }
  2118. }
  2119. override = {}
  2120. actual = config.merge_service_dicts(base, override, V2_1)
  2121. assert actual == base
  2122. def test_merge_depends_on_mixed_syntax(self):
  2123. base = {
  2124. 'image': 'busybox',
  2125. 'depends_on': {
  2126. 'app1': {'condition': 'service_started'},
  2127. 'app2': {'condition': 'service_healthy'}
  2128. }
  2129. }
  2130. override = {
  2131. 'depends_on': ['app3']
  2132. }
  2133. actual = config.merge_service_dicts(base, override, V2_1)
  2134. assert actual == {
  2135. 'image': 'busybox',
  2136. 'depends_on': {
  2137. 'app1': {'condition': 'service_started'},
  2138. 'app2': {'condition': 'service_healthy'},
  2139. 'app3': {'condition': 'service_started'}
  2140. }
  2141. }
  2142. def test_empty_environment_key_allowed(self):
  2143. service_dict = config.load(
  2144. build_config_details(
  2145. {
  2146. 'web': {
  2147. 'build': '.',
  2148. 'environment': {
  2149. 'POSTGRES_PASSWORD': ''
  2150. },
  2151. },
  2152. },
  2153. '.',
  2154. None,
  2155. )
  2156. ).services[0]
  2157. assert service_dict['environment']['POSTGRES_PASSWORD'] == ''
  2158. def test_merge_pid(self):
  2159. # Regression: https://github.com/docker/compose/issues/4184
  2160. base = {
  2161. 'image': 'busybox',
  2162. 'pid': 'host'
  2163. }
  2164. override = {
  2165. 'labels': {'com.docker.compose.test': 'yes'}
  2166. }
  2167. actual = config.merge_service_dicts(base, override, V2_0)
  2168. assert actual == {
  2169. 'image': 'busybox',
  2170. 'pid': 'host',
  2171. 'labels': {'com.docker.compose.test': 'yes'}
  2172. }
  2173. def test_merge_different_secrets(self):
  2174. base = {
  2175. 'image': 'busybox',
  2176. 'secrets': [
  2177. {'source': 'src.txt'}
  2178. ]
  2179. }
  2180. override = {'secrets': ['other-src.txt']}
  2181. actual = config.merge_service_dicts(base, override, V3_1)
  2182. assert secret_sort(actual['secrets']) == secret_sort([
  2183. {'source': 'src.txt'},
  2184. {'source': 'other-src.txt'}
  2185. ])
  2186. def test_merge_secrets_override(self):
  2187. base = {
  2188. 'image': 'busybox',
  2189. 'secrets': ['src.txt'],
  2190. }
  2191. override = {
  2192. 'secrets': [
  2193. {
  2194. 'source': 'src.txt',
  2195. 'target': 'data.txt',
  2196. 'mode': 0o400
  2197. }
  2198. ]
  2199. }
  2200. actual = config.merge_service_dicts(base, override, V3_1)
  2201. assert actual['secrets'] == override['secrets']
  2202. def test_merge_different_configs(self):
  2203. base = {
  2204. 'image': 'busybox',
  2205. 'configs': [
  2206. {'source': 'src.txt'}
  2207. ]
  2208. }
  2209. override = {'configs': ['other-src.txt']}
  2210. actual = config.merge_service_dicts(base, override, V3_3)
  2211. assert secret_sort(actual['configs']) == secret_sort([
  2212. {'source': 'src.txt'},
  2213. {'source': 'other-src.txt'}
  2214. ])
  2215. def test_merge_configs_override(self):
  2216. base = {
  2217. 'image': 'busybox',
  2218. 'configs': ['src.txt'],
  2219. }
  2220. override = {
  2221. 'configs': [
  2222. {
  2223. 'source': 'src.txt',
  2224. 'target': 'data.txt',
  2225. 'mode': 0o400
  2226. }
  2227. ]
  2228. }
  2229. actual = config.merge_service_dicts(base, override, V3_3)
  2230. assert actual['configs'] == override['configs']
  2231. def test_merge_deploy(self):
  2232. base = {
  2233. 'image': 'busybox',
  2234. }
  2235. override = {
  2236. 'deploy': {
  2237. 'mode': 'global',
  2238. 'restart_policy': {
  2239. 'condition': 'on-failure'
  2240. }
  2241. }
  2242. }
  2243. actual = config.merge_service_dicts(base, override, V3_0)
  2244. assert actual['deploy'] == override['deploy']
  2245. def test_merge_deploy_override(self):
  2246. base = {
  2247. 'deploy': {
  2248. 'endpoint_mode': 'vip',
  2249. 'labels': ['com.docker.compose.a=1', 'com.docker.compose.b=2'],
  2250. 'mode': 'replicated',
  2251. 'placement': {
  2252. 'constraints': [
  2253. 'node.role == manager', 'engine.labels.aws == true'
  2254. ],
  2255. 'preferences': [
  2256. {'spread': 'node.labels.zone'}, {'spread': 'x.d.z'}
  2257. ]
  2258. },
  2259. 'replicas': 3,
  2260. 'resources': {
  2261. 'limits': {'cpus': '0.50', 'memory': '50m'},
  2262. 'reservations': {
  2263. 'cpus': '0.1',
  2264. 'generic_resources': [
  2265. {'discrete_resource_spec': {'kind': 'abc', 'value': 123}}
  2266. ],
  2267. 'memory': '15m'
  2268. }
  2269. },
  2270. 'restart_policy': {'condition': 'any', 'delay': '10s'},
  2271. 'update_config': {'delay': '10s', 'max_failure_ratio': 0.3}
  2272. },
  2273. 'image': 'hello-world'
  2274. }
  2275. override = {
  2276. 'deploy': {
  2277. 'labels': {
  2278. 'com.docker.compose.b': '21', 'com.docker.compose.c': '3'
  2279. },
  2280. 'placement': {
  2281. 'constraints': ['node.role == worker', 'engine.labels.dev == true'],
  2282. 'preferences': [{'spread': 'node.labels.zone'}, {'spread': 'x.d.s'}]
  2283. },
  2284. 'resources': {
  2285. 'limits': {'memory': '200m'},
  2286. 'reservations': {
  2287. 'cpus': '0.78',
  2288. 'generic_resources': [
  2289. {'discrete_resource_spec': {'kind': 'abc', 'value': 134}},
  2290. {'discrete_resource_spec': {'kind': 'xyz', 'value': 0.1}}
  2291. ]
  2292. }
  2293. },
  2294. 'restart_policy': {'condition': 'on-failure', 'max_attempts': 42},
  2295. 'update_config': {'max_failure_ratio': 0.712, 'parallelism': 4}
  2296. }
  2297. }
  2298. actual = config.merge_service_dicts(base, override, V3_5)
  2299. assert actual['deploy'] == {
  2300. 'mode': 'replicated',
  2301. 'endpoint_mode': 'vip',
  2302. 'labels': {
  2303. 'com.docker.compose.a': '1',
  2304. 'com.docker.compose.b': '21',
  2305. 'com.docker.compose.c': '3'
  2306. },
  2307. 'placement': {
  2308. 'constraints': [
  2309. 'engine.labels.aws == true', 'engine.labels.dev == true',
  2310. 'node.role == manager', 'node.role == worker'
  2311. ],
  2312. 'preferences': [
  2313. {'spread': 'node.labels.zone'}, {'spread': 'x.d.s'}, {'spread': 'x.d.z'}
  2314. ]
  2315. },
  2316. 'replicas': 3,
  2317. 'resources': {
  2318. 'limits': {'cpus': '0.50', 'memory': '200m'},
  2319. 'reservations': {
  2320. 'cpus': '0.78',
  2321. 'memory': '15m',
  2322. 'generic_resources': [
  2323. {'discrete_resource_spec': {'kind': 'abc', 'value': 134}},
  2324. {'discrete_resource_spec': {'kind': 'xyz', 'value': 0.1}},
  2325. ]
  2326. }
  2327. },
  2328. 'restart_policy': {
  2329. 'condition': 'on-failure',
  2330. 'delay': '10s',
  2331. 'max_attempts': 42,
  2332. },
  2333. 'update_config': {
  2334. 'max_failure_ratio': 0.712,
  2335. 'delay': '10s',
  2336. 'parallelism': 4
  2337. }
  2338. }
  2339. def test_merge_credential_spec(self):
  2340. base = {
  2341. 'image': 'bb',
  2342. 'credential_spec': {
  2343. 'file': '/hello-world',
  2344. }
  2345. }
  2346. override = {
  2347. 'credential_spec': {
  2348. 'registry': 'revolution.com',
  2349. }
  2350. }
  2351. actual = config.merge_service_dicts(base, override, V3_3)
  2352. assert actual['credential_spec'] == override['credential_spec']
  2353. def test_merge_scale(self):
  2354. base = {
  2355. 'image': 'bar',
  2356. 'scale': 2,
  2357. }
  2358. override = {
  2359. 'scale': 4,
  2360. }
  2361. actual = config.merge_service_dicts(base, override, V2_2)
  2362. assert actual == {'image': 'bar', 'scale': 4}
  2363. def test_merge_blkio_config(self):
  2364. base = {
  2365. 'image': 'bar',
  2366. 'blkio_config': {
  2367. 'weight': 300,
  2368. 'weight_device': [
  2369. {'path': '/dev/sda1', 'weight': 200}
  2370. ],
  2371. 'device_read_iops': [
  2372. {'path': '/dev/sda1', 'rate': 300}
  2373. ],
  2374. 'device_write_iops': [
  2375. {'path': '/dev/sda1', 'rate': 1000}
  2376. ]
  2377. }
  2378. }
  2379. override = {
  2380. 'blkio_config': {
  2381. 'weight': 450,
  2382. 'weight_device': [
  2383. {'path': '/dev/sda2', 'weight': 400}
  2384. ],
  2385. 'device_read_iops': [
  2386. {'path': '/dev/sda1', 'rate': 2000}
  2387. ],
  2388. 'device_read_bps': [
  2389. {'path': '/dev/sda1', 'rate': 1024}
  2390. ]
  2391. }
  2392. }
  2393. actual = config.merge_service_dicts(base, override, V2_2)
  2394. assert actual == {
  2395. 'image': 'bar',
  2396. 'blkio_config': {
  2397. 'weight': override['blkio_config']['weight'],
  2398. 'weight_device': (
  2399. base['blkio_config']['weight_device'] +
  2400. override['blkio_config']['weight_device']
  2401. ),
  2402. 'device_read_iops': override['blkio_config']['device_read_iops'],
  2403. 'device_read_bps': override['blkio_config']['device_read_bps'],
  2404. 'device_write_iops': base['blkio_config']['device_write_iops']
  2405. }
  2406. }
  2407. def test_merge_extra_hosts(self):
  2408. base = {
  2409. 'image': 'bar',
  2410. 'extra_hosts': {
  2411. 'foo': '1.2.3.4',
  2412. }
  2413. }
  2414. override = {
  2415. 'extra_hosts': ['bar:5.6.7.8', 'foo:127.0.0.1']
  2416. }
  2417. actual = config.merge_service_dicts(base, override, V2_0)
  2418. assert actual['extra_hosts'] == {
  2419. 'foo': '127.0.0.1',
  2420. 'bar': '5.6.7.8',
  2421. }
  2422. def test_merge_healthcheck_config(self):
  2423. base = {
  2424. 'image': 'bar',
  2425. 'healthcheck': {
  2426. 'start_period': 1000,
  2427. 'interval': 3000,
  2428. 'test': ['true']
  2429. }
  2430. }
  2431. override = {
  2432. 'healthcheck': {
  2433. 'interval': 5000,
  2434. 'timeout': 10000,
  2435. 'test': ['echo', 'OK'],
  2436. }
  2437. }
  2438. actual = config.merge_service_dicts(base, override, V2_3)
  2439. assert actual['healthcheck'] == {
  2440. 'start_period': base['healthcheck']['start_period'],
  2441. 'test': override['healthcheck']['test'],
  2442. 'interval': override['healthcheck']['interval'],
  2443. 'timeout': override['healthcheck']['timeout'],
  2444. }
  2445. def test_merge_healthcheck_override_disables(self):
  2446. base = {
  2447. 'image': 'bar',
  2448. 'healthcheck': {
  2449. 'start_period': 1000,
  2450. 'interval': 3000,
  2451. 'timeout': 2000,
  2452. 'retries': 3,
  2453. 'test': ['true']
  2454. }
  2455. }
  2456. override = {
  2457. 'healthcheck': {
  2458. 'disabled': True
  2459. }
  2460. }
  2461. actual = config.merge_service_dicts(base, override, V2_3)
  2462. assert actual['healthcheck'] == {'disabled': True}
  2463. def test_merge_healthcheck_override_enables(self):
  2464. base = {
  2465. 'image': 'bar',
  2466. 'healthcheck': {
  2467. 'disabled': True
  2468. }
  2469. }
  2470. override = {
  2471. 'healthcheck': {
  2472. 'disabled': False,
  2473. 'start_period': 1000,
  2474. 'interval': 3000,
  2475. 'timeout': 2000,
  2476. 'retries': 3,
  2477. 'test': ['true']
  2478. }
  2479. }
  2480. actual = config.merge_service_dicts(base, override, V2_3)
  2481. assert actual['healthcheck'] == override['healthcheck']
  2482. def test_merge_device_cgroup_rules(self):
  2483. base = {
  2484. 'image': 'bar',
  2485. 'device_cgroup_rules': ['c 7:128 rwm', 'x 3:244 rw']
  2486. }
  2487. override = {
  2488. 'device_cgroup_rules': ['c 7:128 rwm', 'f 0:128 n']
  2489. }
  2490. actual = config.merge_service_dicts(base, override, V2_3)
  2491. assert sorted(actual['device_cgroup_rules']) == sorted(
  2492. ['c 7:128 rwm', 'x 3:244 rw', 'f 0:128 n']
  2493. )
  2494. def test_merge_isolation(self):
  2495. base = {
  2496. 'image': 'bar',
  2497. 'isolation': 'default',
  2498. }
  2499. override = {
  2500. 'isolation': 'hyperv',
  2501. }
  2502. actual = config.merge_service_dicts(base, override, V2_3)
  2503. assert actual == {
  2504. 'image': 'bar',
  2505. 'isolation': 'hyperv',
  2506. }
  2507. def test_merge_storage_opt(self):
  2508. base = {
  2509. 'image': 'bar',
  2510. 'storage_opt': {
  2511. 'size': '1G',
  2512. 'readonly': 'false',
  2513. }
  2514. }
  2515. override = {
  2516. 'storage_opt': {
  2517. 'size': '2G',
  2518. 'encryption': 'aes',
  2519. }
  2520. }
  2521. actual = config.merge_service_dicts(base, override, V2_3)
  2522. assert actual['storage_opt'] == {
  2523. 'size': '2G',
  2524. 'readonly': 'false',
  2525. 'encryption': 'aes',
  2526. }
  2527. def test_external_volume_config(self):
  2528. config_details = build_config_details({
  2529. 'version': '2',
  2530. 'services': {
  2531. 'bogus': {'image': 'busybox'}
  2532. },
  2533. 'volumes': {
  2534. 'ext': {'external': True},
  2535. 'ext2': {'external': {'name': 'aliased'}}
  2536. }
  2537. })
  2538. config_result = config.load(config_details)
  2539. volumes = config_result.volumes
  2540. assert 'ext' in volumes
  2541. assert volumes['ext']['external'] is True
  2542. assert 'ext2' in volumes
  2543. assert volumes['ext2']['external']['name'] == 'aliased'
  2544. def test_external_volume_invalid_config(self):
  2545. config_details = build_config_details({
  2546. 'version': '2',
  2547. 'services': {
  2548. 'bogus': {'image': 'busybox'}
  2549. },
  2550. 'volumes': {
  2551. 'ext': {'external': True, 'driver': 'foo'}
  2552. }
  2553. })
  2554. with pytest.raises(ConfigurationError):
  2555. config.load(config_details)
  2556. def test_depends_on_orders_services(self):
  2557. config_details = build_config_details({
  2558. 'version': '2',
  2559. 'services': {
  2560. 'one': {'image': 'busybox', 'depends_on': ['three', 'two']},
  2561. 'two': {'image': 'busybox', 'depends_on': ['three']},
  2562. 'three': {'image': 'busybox'},
  2563. },
  2564. })
  2565. actual = config.load(config_details)
  2566. assert (
  2567. [service['name'] for service in actual.services] ==
  2568. ['three', 'two', 'one']
  2569. )
  2570. def test_depends_on_unknown_service_errors(self):
  2571. config_details = build_config_details({
  2572. 'version': '2',
  2573. 'services': {
  2574. 'one': {'image': 'busybox', 'depends_on': ['three']},
  2575. },
  2576. })
  2577. with pytest.raises(ConfigurationError) as exc:
  2578. config.load(config_details)
  2579. assert "Service 'one' depends on service 'three'" in exc.exconly()
  2580. def test_linked_service_is_undefined(self):
  2581. with pytest.raises(ConfigurationError):
  2582. config.load(
  2583. build_config_details({
  2584. 'version': '2',
  2585. 'services': {
  2586. 'web': {'image': 'busybox', 'links': ['db:db']},
  2587. },
  2588. })
  2589. )
  2590. def test_load_dockerfile_without_context(self):
  2591. config_details = build_config_details({
  2592. 'version': '2',
  2593. 'services': {
  2594. 'one': {'build': {'dockerfile': 'Dockerfile.foo'}},
  2595. },
  2596. })
  2597. with pytest.raises(ConfigurationError) as exc:
  2598. config.load(config_details)
  2599. assert 'has neither an image nor a build context' in exc.exconly()
  2600. def test_load_secrets(self):
  2601. base_file = config.ConfigFile(
  2602. 'base.yaml',
  2603. {
  2604. 'version': '3.1',
  2605. 'services': {
  2606. 'web': {
  2607. 'image': 'example/web',
  2608. 'secrets': [
  2609. 'one',
  2610. {
  2611. 'source': 'source',
  2612. 'target': 'target',
  2613. 'uid': '100',
  2614. 'gid': '200',
  2615. 'mode': 0o777,
  2616. },
  2617. ],
  2618. },
  2619. },
  2620. 'secrets': {
  2621. 'one': {'file': 'secret.txt'},
  2622. },
  2623. })
  2624. details = config.ConfigDetails('.', [base_file])
  2625. service_dicts = config.load(details).services
  2626. expected = [
  2627. {
  2628. 'name': 'web',
  2629. 'image': 'example/web',
  2630. 'secrets': [
  2631. types.ServiceSecret('one', None, None, None, None, None),
  2632. types.ServiceSecret('source', 'target', '100', '200', 0o777, None),
  2633. ],
  2634. },
  2635. ]
  2636. assert service_sort(service_dicts) == service_sort(expected)
  2637. def test_load_secrets_multi_file(self):
  2638. base_file = config.ConfigFile(
  2639. 'base.yaml',
  2640. {
  2641. 'version': '3.1',
  2642. 'services': {
  2643. 'web': {
  2644. 'image': 'example/web',
  2645. 'secrets': ['one'],
  2646. },
  2647. },
  2648. 'secrets': {
  2649. 'one': {'file': 'secret.txt'},
  2650. },
  2651. })
  2652. override_file = config.ConfigFile(
  2653. 'base.yaml',
  2654. {
  2655. 'version': '3.1',
  2656. 'services': {
  2657. 'web': {
  2658. 'secrets': [
  2659. {
  2660. 'source': 'source',
  2661. 'target': 'target',
  2662. 'uid': '100',
  2663. 'gid': '200',
  2664. 'mode': 0o777,
  2665. },
  2666. ],
  2667. },
  2668. },
  2669. })
  2670. details = config.ConfigDetails('.', [base_file, override_file])
  2671. service_dicts = config.load(details).services
  2672. expected = [
  2673. {
  2674. 'name': 'web',
  2675. 'image': 'example/web',
  2676. 'secrets': [
  2677. types.ServiceSecret('one', None, None, None, None, None),
  2678. types.ServiceSecret('source', 'target', '100', '200', 0o777, None),
  2679. ],
  2680. },
  2681. ]
  2682. assert service_sort(service_dicts) == service_sort(expected)
  2683. def test_load_configs(self):
  2684. base_file = config.ConfigFile(
  2685. 'base.yaml',
  2686. {
  2687. 'version': '3.3',
  2688. 'services': {
  2689. 'web': {
  2690. 'image': 'example/web',
  2691. 'configs': [
  2692. 'one',
  2693. {
  2694. 'source': 'source',
  2695. 'target': 'target',
  2696. 'uid': '100',
  2697. 'gid': '200',
  2698. 'mode': 0o777,
  2699. },
  2700. ],
  2701. },
  2702. },
  2703. 'configs': {
  2704. 'one': {'file': 'secret.txt'},
  2705. },
  2706. })
  2707. details = config.ConfigDetails('.', [base_file])
  2708. service_dicts = config.load(details).services
  2709. expected = [
  2710. {
  2711. 'name': 'web',
  2712. 'image': 'example/web',
  2713. 'configs': [
  2714. types.ServiceConfig('one', None, None, None, None, None),
  2715. types.ServiceConfig('source', 'target', '100', '200', 0o777, None),
  2716. ],
  2717. },
  2718. ]
  2719. assert service_sort(service_dicts) == service_sort(expected)
  2720. def test_load_configs_multi_file(self):
  2721. base_file = config.ConfigFile(
  2722. 'base.yaml',
  2723. {
  2724. 'version': '3.3',
  2725. 'services': {
  2726. 'web': {
  2727. 'image': 'example/web',
  2728. 'configs': ['one'],
  2729. },
  2730. },
  2731. 'configs': {
  2732. 'one': {'file': 'secret.txt'},
  2733. },
  2734. })
  2735. override_file = config.ConfigFile(
  2736. 'base.yaml',
  2737. {
  2738. 'version': '3.3',
  2739. 'services': {
  2740. 'web': {
  2741. 'configs': [
  2742. {
  2743. 'source': 'source',
  2744. 'target': 'target',
  2745. 'uid': '100',
  2746. 'gid': '200',
  2747. 'mode': 0o777,
  2748. },
  2749. ],
  2750. },
  2751. },
  2752. })
  2753. details = config.ConfigDetails('.', [base_file, override_file])
  2754. service_dicts = config.load(details).services
  2755. expected = [
  2756. {
  2757. 'name': 'web',
  2758. 'image': 'example/web',
  2759. 'configs': [
  2760. types.ServiceConfig('one', None, None, None, None, None),
  2761. types.ServiceConfig('source', 'target', '100', '200', 0o777, None),
  2762. ],
  2763. },
  2764. ]
  2765. assert service_sort(service_dicts) == service_sort(expected)
  2766. def test_config_convertible_label_types(self):
  2767. config_details = build_config_details(
  2768. {
  2769. 'version': '3.5',
  2770. 'services': {
  2771. 'web': {
  2772. 'build': {
  2773. 'labels': {'testbuild': True},
  2774. 'context': os.getcwd()
  2775. },
  2776. 'labels': {
  2777. "key": 12345
  2778. }
  2779. },
  2780. },
  2781. 'networks': {
  2782. 'foo': {
  2783. 'labels': {'network.ips.max': 1023}
  2784. }
  2785. },
  2786. 'volumes': {
  2787. 'foo': {
  2788. 'labels': {'volume.is_readonly': False}
  2789. }
  2790. },
  2791. 'secrets': {
  2792. 'foo': {
  2793. 'labels': {'secret.data.expires': 1546282120}
  2794. }
  2795. },
  2796. 'configs': {
  2797. 'foo': {
  2798. 'labels': {'config.data.correction.value': -0.1412}
  2799. }
  2800. }
  2801. }
  2802. )
  2803. loaded_config = config.load(config_details)
  2804. assert loaded_config.services[0]['build']['labels'] == {'testbuild': 'True'}
  2805. assert loaded_config.services[0]['labels'] == {'key': '12345'}
  2806. assert loaded_config.networks['foo']['labels']['network.ips.max'] == '1023'
  2807. assert loaded_config.volumes['foo']['labels']['volume.is_readonly'] == 'False'
  2808. assert loaded_config.secrets['foo']['labels']['secret.data.expires'] == '1546282120'
  2809. assert loaded_config.configs['foo']['labels']['config.data.correction.value'] == '-0.1412'
  2810. def test_config_invalid_label_types(self):
  2811. config_details = build_config_details({
  2812. 'version': '2.3',
  2813. 'volumes': {
  2814. 'foo': {'labels': [1, 2, 3]}
  2815. }
  2816. })
  2817. with pytest.raises(ConfigurationError):
  2818. config.load(config_details)
  2819. def test_service_volume_invalid_config(self):
  2820. config_details = build_config_details(
  2821. {
  2822. 'version': '3.2',
  2823. 'services': {
  2824. 'web': {
  2825. 'build': {
  2826. 'context': '.',
  2827. 'args': None,
  2828. },
  2829. 'volumes': [
  2830. {
  2831. "type": "volume",
  2832. "source": "/data",
  2833. "garbage": {
  2834. "and": "error"
  2835. }
  2836. }
  2837. ]
  2838. }
  2839. }
  2840. }
  2841. )
  2842. with pytest.raises(ConfigurationError) as exc:
  2843. config.load(config_details)
  2844. assert "services.web.volumes contains unsupported option: 'garbage'" in exc.exconly()
  2845. def test_config_valid_service_label_validation(self):
  2846. config_details = build_config_details(
  2847. {
  2848. 'version': '3.5',
  2849. 'services': {
  2850. 'web': {
  2851. 'image': 'busybox',
  2852. 'labels': {
  2853. "key": "string"
  2854. }
  2855. },
  2856. },
  2857. }
  2858. )
  2859. config.load(config_details)
  2860. def test_config_duplicate_mount_points(self):
  2861. config1 = build_config_details(
  2862. {
  2863. 'version': '3.5',
  2864. 'services': {
  2865. 'web': {
  2866. 'image': 'busybox',
  2867. 'volumes': ['/tmp/foo:/tmp/foo', '/tmp/foo:/tmp/foo:rw']
  2868. }
  2869. }
  2870. }
  2871. )
  2872. config2 = build_config_details(
  2873. {
  2874. 'version': '3.5',
  2875. 'services': {
  2876. 'web': {
  2877. 'image': 'busybox',
  2878. 'volumes': ['/x:/y', '/z:/y']
  2879. }
  2880. }
  2881. }
  2882. )
  2883. with self.assertRaises(ConfigurationError) as e:
  2884. config.load(config1)
  2885. self.assertEquals(str(e.exception), 'Duplicate mount points: [%s]' % (
  2886. ', '.join(['/tmp/foo:/tmp/foo:rw']*2)))
  2887. with self.assertRaises(ConfigurationError) as e:
  2888. config.load(config2)
  2889. self.assertEquals(str(e.exception), 'Duplicate mount points: [%s]' % (
  2890. ', '.join(['/x:/y:rw', '/z:/y:rw'])))
  2891. class NetworkModeTest(unittest.TestCase):
  2892. def test_network_mode_standard(self):
  2893. config_data = config.load(build_config_details({
  2894. 'version': '2',
  2895. 'services': {
  2896. 'web': {
  2897. 'image': 'busybox',
  2898. 'command': "top",
  2899. 'network_mode': 'bridge',
  2900. },
  2901. },
  2902. }))
  2903. assert config_data.services[0]['network_mode'] == 'bridge'
  2904. def test_network_mode_standard_v1(self):
  2905. config_data = config.load(build_config_details({
  2906. 'web': {
  2907. 'image': 'busybox',
  2908. 'command': "top",
  2909. 'net': 'bridge',
  2910. },
  2911. }))
  2912. assert config_data.services[0]['network_mode'] == 'bridge'
  2913. assert 'net' not in config_data.services[0]
  2914. def test_network_mode_container(self):
  2915. config_data = config.load(build_config_details({
  2916. 'version': '2',
  2917. 'services': {
  2918. 'web': {
  2919. 'image': 'busybox',
  2920. 'command': "top",
  2921. 'network_mode': 'container:foo',
  2922. },
  2923. },
  2924. }))
  2925. assert config_data.services[0]['network_mode'] == 'container:foo'
  2926. def test_network_mode_container_v1(self):
  2927. config_data = config.load(build_config_details({
  2928. 'web': {
  2929. 'image': 'busybox',
  2930. 'command': "top",
  2931. 'net': 'container:foo',
  2932. },
  2933. }))
  2934. assert config_data.services[0]['network_mode'] == 'container:foo'
  2935. def test_network_mode_service(self):
  2936. config_data = config.load(build_config_details({
  2937. 'version': '2',
  2938. 'services': {
  2939. 'web': {
  2940. 'image': 'busybox',
  2941. 'command': "top",
  2942. 'network_mode': 'service:foo',
  2943. },
  2944. 'foo': {
  2945. 'image': 'busybox',
  2946. 'command': "top",
  2947. },
  2948. },
  2949. }))
  2950. assert config_data.services[1]['network_mode'] == 'service:foo'
  2951. def test_network_mode_service_v1(self):
  2952. config_data = config.load(build_config_details({
  2953. 'web': {
  2954. 'image': 'busybox',
  2955. 'command': "top",
  2956. 'net': 'container:foo',
  2957. },
  2958. 'foo': {
  2959. 'image': 'busybox',
  2960. 'command': "top",
  2961. },
  2962. }))
  2963. assert config_data.services[1]['network_mode'] == 'service:foo'
  2964. def test_network_mode_service_nonexistent(self):
  2965. with pytest.raises(ConfigurationError) as excinfo:
  2966. config.load(build_config_details({
  2967. 'version': '2',
  2968. 'services': {
  2969. 'web': {
  2970. 'image': 'busybox',
  2971. 'command': "top",
  2972. 'network_mode': 'service:foo',
  2973. },
  2974. },
  2975. }))
  2976. assert "service 'foo' which is undefined" in excinfo.exconly()
  2977. def test_network_mode_plus_networks_is_invalid(self):
  2978. with pytest.raises(ConfigurationError) as excinfo:
  2979. config.load(build_config_details({
  2980. 'version': '2',
  2981. 'services': {
  2982. 'web': {
  2983. 'image': 'busybox',
  2984. 'command': "top",
  2985. 'network_mode': 'bridge',
  2986. 'networks': ['front'],
  2987. },
  2988. },
  2989. 'networks': {
  2990. 'front': None,
  2991. }
  2992. }))
  2993. assert "'network_mode' and 'networks' cannot be combined" in excinfo.exconly()
  2994. class PortsTest(unittest.TestCase):
  2995. INVALID_PORTS_TYPES = [
  2996. {"1": "8000"},
  2997. False,
  2998. "8000",
  2999. 8000,
  3000. ]
  3001. NON_UNIQUE_SINGLE_PORTS = [
  3002. ["8000", "8000"],
  3003. ]
  3004. INVALID_PORT_MAPPINGS = [
  3005. ["8000-8004:8000-8002"],
  3006. ["4242:4242-4244"],
  3007. ]
  3008. VALID_SINGLE_PORTS = [
  3009. ["8000"],
  3010. ["8000/tcp"],
  3011. ["8000", "9000"],
  3012. [8000],
  3013. [8000, 9000],
  3014. ]
  3015. VALID_PORT_MAPPINGS = [
  3016. ["8000:8050"],
  3017. ["49153-49154:3002-3003"],
  3018. ]
  3019. def test_config_invalid_ports_type_validation(self):
  3020. for invalid_ports in self.INVALID_PORTS_TYPES:
  3021. with pytest.raises(ConfigurationError) as exc:
  3022. self.check_config({'ports': invalid_ports})
  3023. assert "contains an invalid type" in exc.value.msg
  3024. def test_config_non_unique_ports_validation(self):
  3025. for invalid_ports in self.NON_UNIQUE_SINGLE_PORTS:
  3026. with pytest.raises(ConfigurationError) as exc:
  3027. self.check_config({'ports': invalid_ports})
  3028. assert "non-unique" in exc.value.msg
  3029. def test_config_invalid_ports_format_validation(self):
  3030. for invalid_ports in self.INVALID_PORT_MAPPINGS:
  3031. with pytest.raises(ConfigurationError) as exc:
  3032. self.check_config({'ports': invalid_ports})
  3033. assert "Port ranges don't match in length" in exc.value.msg
  3034. def test_config_valid_ports_format_validation(self):
  3035. for valid_ports in self.VALID_SINGLE_PORTS + self.VALID_PORT_MAPPINGS:
  3036. self.check_config({'ports': valid_ports})
  3037. def test_config_invalid_expose_type_validation(self):
  3038. for invalid_expose in self.INVALID_PORTS_TYPES:
  3039. with pytest.raises(ConfigurationError) as exc:
  3040. self.check_config({'expose': invalid_expose})
  3041. assert "contains an invalid type" in exc.value.msg
  3042. def test_config_non_unique_expose_validation(self):
  3043. for invalid_expose in self.NON_UNIQUE_SINGLE_PORTS:
  3044. with pytest.raises(ConfigurationError) as exc:
  3045. self.check_config({'expose': invalid_expose})
  3046. assert "non-unique" in exc.value.msg
  3047. def test_config_invalid_expose_format_validation(self):
  3048. # Valid port mappings ARE NOT valid 'expose' entries
  3049. for invalid_expose in self.INVALID_PORT_MAPPINGS + self.VALID_PORT_MAPPINGS:
  3050. with pytest.raises(ConfigurationError) as exc:
  3051. self.check_config({'expose': invalid_expose})
  3052. assert "should be of the format" in exc.value.msg
  3053. def test_config_valid_expose_format_validation(self):
  3054. # Valid single ports ARE valid 'expose' entries
  3055. for valid_expose in self.VALID_SINGLE_PORTS:
  3056. self.check_config({'expose': valid_expose})
  3057. def check_config(self, cfg):
  3058. config.load(
  3059. build_config_details({
  3060. 'version': '2.3',
  3061. 'services': {
  3062. 'web': dict(image='busybox', **cfg)
  3063. },
  3064. }, 'working_dir', 'filename.yml')
  3065. )
  3066. class SubnetTest(unittest.TestCase):
  3067. INVALID_SUBNET_TYPES = [
  3068. None,
  3069. False,
  3070. 10,
  3071. ]
  3072. INVALID_SUBNET_MAPPINGS = [
  3073. "",
  3074. "192.168.0.1/sdfsdfs",
  3075. "192.168.0.1/",
  3076. "192.168.0.1/33",
  3077. "192.168.0.1/01",
  3078. "192.168.0.1",
  3079. "fe80:0000:0000:0000:0204:61ff:fe9d:f156/sdfsdfs",
  3080. "fe80:0000:0000:0000:0204:61ff:fe9d:f156/",
  3081. "fe80:0000:0000:0000:0204:61ff:fe9d:f156/129",
  3082. "fe80:0000:0000:0000:0204:61ff:fe9d:f156/01",
  3083. "fe80:0000:0000:0000:0204:61ff:fe9d:f156",
  3084. "ge80:0000:0000:0000:0204:61ff:fe9d:f156/128",
  3085. "192.168.0.1/31/31",
  3086. ]
  3087. VALID_SUBNET_MAPPINGS = [
  3088. "192.168.0.1/0",
  3089. "192.168.0.1/32",
  3090. "fe80:0000:0000:0000:0204:61ff:fe9d:f156/0",
  3091. "fe80:0000:0000:0000:0204:61ff:fe9d:f156/128",
  3092. "1:2:3:4:5:6:7:8/0",
  3093. "1::/0",
  3094. "1:2:3:4:5:6:7::/0",
  3095. "1::8/0",
  3096. "1:2:3:4:5:6::8/0",
  3097. "::/0",
  3098. "::8/0",
  3099. "::2:3:4:5:6:7:8/0",
  3100. "fe80::7:8%eth0/0",
  3101. "fe80::7:8%1/0",
  3102. "::255.255.255.255/0",
  3103. "::ffff:255.255.255.255/0",
  3104. "::ffff:0:255.255.255.255/0",
  3105. "2001:db8:3:4::192.0.2.33/0",
  3106. "64:ff9b::192.0.2.33/0",
  3107. ]
  3108. def test_config_invalid_subnet_type_validation(self):
  3109. for invalid_subnet in self.INVALID_SUBNET_TYPES:
  3110. with pytest.raises(ConfigurationError) as exc:
  3111. self.check_config(invalid_subnet)
  3112. assert "contains an invalid type" in exc.value.msg
  3113. def test_config_invalid_subnet_format_validation(self):
  3114. for invalid_subnet in self.INVALID_SUBNET_MAPPINGS:
  3115. with pytest.raises(ConfigurationError) as exc:
  3116. self.check_config(invalid_subnet)
  3117. assert "should use the CIDR format" in exc.value.msg
  3118. def test_config_valid_subnet_format_validation(self):
  3119. for valid_subnet in self.VALID_SUBNET_MAPPINGS:
  3120. self.check_config(valid_subnet)
  3121. def check_config(self, subnet):
  3122. config.load(
  3123. build_config_details({
  3124. 'version': '3.5',
  3125. 'services': {
  3126. 'web': {
  3127. 'image': 'busybox'
  3128. }
  3129. },
  3130. 'networks': {
  3131. 'default': {
  3132. 'ipam': {
  3133. 'config': [
  3134. {
  3135. 'subnet': subnet
  3136. }
  3137. ],
  3138. 'driver': 'default'
  3139. }
  3140. }
  3141. }
  3142. })
  3143. )
  3144. class InterpolationTest(unittest.TestCase):
  3145. @mock.patch.dict(os.environ)
  3146. def test_config_file_with_environment_file(self):
  3147. project_dir = 'tests/fixtures/default-env-file'
  3148. service_dicts = config.load(
  3149. config.find(
  3150. project_dir, None, Environment.from_env_file(project_dir)
  3151. )
  3152. ).services
  3153. assert service_dicts[0] == {
  3154. 'name': 'web',
  3155. 'image': 'alpine:latest',
  3156. 'ports': [
  3157. types.ServicePort.parse('5643')[0],
  3158. types.ServicePort.parse('9999')[0]
  3159. ],
  3160. 'command': 'true'
  3161. }
  3162. @mock.patch.dict(os.environ)
  3163. def test_config_file_with_options_environment_file(self):
  3164. project_dir = 'tests/fixtures/default-env-file'
  3165. service_dicts = config.load(
  3166. config.find(
  3167. project_dir, None, Environment.from_env_file(project_dir, '.env2')
  3168. )
  3169. ).services
  3170. assert service_dicts[0] == {
  3171. 'name': 'web',
  3172. 'image': 'alpine:latest',
  3173. 'ports': [
  3174. types.ServicePort.parse('5644')[0],
  3175. types.ServicePort.parse('9998')[0]
  3176. ],
  3177. 'command': 'false'
  3178. }
  3179. @mock.patch.dict(os.environ)
  3180. def test_config_file_with_environment_variable(self):
  3181. project_dir = 'tests/fixtures/environment-interpolation'
  3182. os.environ.update(
  3183. IMAGE="busybox",
  3184. HOST_PORT="80",
  3185. LABEL_VALUE="myvalue",
  3186. )
  3187. service_dicts = config.load(
  3188. config.find(
  3189. project_dir, None, Environment.from_env_file(project_dir)
  3190. )
  3191. ).services
  3192. assert service_dicts == [
  3193. {
  3194. 'name': 'web',
  3195. 'image': 'busybox',
  3196. 'ports': types.ServicePort.parse('80:8000'),
  3197. 'labels': {'mylabel': 'myvalue'},
  3198. 'hostname': 'host-',
  3199. 'command': '${ESCAPED}',
  3200. }
  3201. ]
  3202. @mock.patch.dict(os.environ)
  3203. def test_config_file_with_environment_variable_with_defaults(self):
  3204. project_dir = 'tests/fixtures/environment-interpolation-with-defaults'
  3205. os.environ.update(
  3206. IMAGE="busybox",
  3207. )
  3208. service_dicts = config.load(
  3209. config.find(
  3210. project_dir, None, Environment.from_env_file(project_dir)
  3211. )
  3212. ).services
  3213. assert service_dicts == [
  3214. {
  3215. 'name': 'web',
  3216. 'image': 'busybox',
  3217. 'ports': types.ServicePort.parse('80:8000'),
  3218. 'hostname': 'host-',
  3219. }
  3220. ]
  3221. @mock.patch.dict(os.environ)
  3222. def test_unset_variable_produces_warning(self):
  3223. os.environ.pop('FOO', None)
  3224. os.environ.pop('BAR', None)
  3225. config_details = build_config_details(
  3226. {
  3227. 'web': {
  3228. 'image': '${FOO}',
  3229. 'command': '${BAR}',
  3230. 'container_name': '${BAR}',
  3231. },
  3232. },
  3233. '.',
  3234. None,
  3235. )
  3236. with mock.patch('compose.config.environment.log') as log:
  3237. config.load(config_details)
  3238. assert 2 == log.warning.call_count
  3239. warnings = sorted(args[0][0] for args in log.warning.call_args_list)
  3240. assert 'BAR' in warnings[0]
  3241. assert 'FOO' in warnings[1]
  3242. def test_compatibility_mode_warnings(self):
  3243. config_details = build_config_details({
  3244. 'version': '3.5',
  3245. 'services': {
  3246. 'web': {
  3247. 'deploy': {
  3248. 'labels': ['abc=def'],
  3249. 'endpoint_mode': 'dnsrr',
  3250. 'update_config': {'max_failure_ratio': 0.4},
  3251. 'placement': {'constraints': ['node.id==deadbeef']},
  3252. 'resources': {
  3253. 'reservations': {'cpus': '0.2'}
  3254. },
  3255. 'restart_policy': {
  3256. 'delay': '2s',
  3257. 'window': '12s'
  3258. }
  3259. },
  3260. 'image': 'busybox'
  3261. }
  3262. }
  3263. })
  3264. with mock.patch('compose.config.config.log') as log:
  3265. config.load(config_details, compatibility=True)
  3266. assert log.warning.call_count == 1
  3267. warn_message = log.warning.call_args[0][0]
  3268. assert warn_message.startswith(
  3269. 'The following deploy sub-keys are not supported in compatibility mode'
  3270. )
  3271. assert 'labels' in warn_message
  3272. assert 'endpoint_mode' in warn_message
  3273. assert 'update_config' in warn_message
  3274. assert 'placement' in warn_message
  3275. assert 'resources.reservations.cpus' in warn_message
  3276. assert 'restart_policy.delay' in warn_message
  3277. assert 'restart_policy.window' in warn_message
  3278. def test_compatibility_mode_load(self):
  3279. config_details = build_config_details({
  3280. 'version': '3.5',
  3281. 'services': {
  3282. 'foo': {
  3283. 'image': 'alpine:3.10.1',
  3284. 'deploy': {
  3285. 'replicas': 3,
  3286. 'restart_policy': {
  3287. 'condition': 'any',
  3288. 'max_attempts': 7,
  3289. },
  3290. 'resources': {
  3291. 'limits': {'memory': '300M', 'cpus': '0.7'},
  3292. 'reservations': {'memory': '100M'},
  3293. },
  3294. },
  3295. 'credential_spec': {
  3296. 'file': 'spec.json'
  3297. },
  3298. },
  3299. },
  3300. })
  3301. with mock.patch('compose.config.config.log') as log:
  3302. cfg = config.load(config_details, compatibility=True)
  3303. assert log.warning.call_count == 0
  3304. service_dict = cfg.services[0]
  3305. assert service_dict == {
  3306. 'image': 'alpine:3.10.1',
  3307. 'scale': 3,
  3308. 'restart': {'MaximumRetryCount': 7, 'Name': 'always'},
  3309. 'mem_limit': '300M',
  3310. 'mem_reservation': '100M',
  3311. 'cpus': 0.7,
  3312. 'name': 'foo',
  3313. 'security_opt': ['credentialspec=file://spec.json'],
  3314. }
  3315. @mock.patch.dict(os.environ)
  3316. def test_invalid_interpolation(self):
  3317. with pytest.raises(config.ConfigurationError) as cm:
  3318. config.load(
  3319. build_config_details(
  3320. {'web': {'image': '${'}},
  3321. 'working_dir',
  3322. 'filename.yml'
  3323. )
  3324. )
  3325. assert 'Invalid' in cm.value.msg
  3326. assert 'for "image" option' in cm.value.msg
  3327. assert 'in service "web"' in cm.value.msg
  3328. assert '"${"' in cm.value.msg
  3329. @mock.patch.dict(os.environ)
  3330. def test_interpolation_secrets_section(self):
  3331. os.environ['FOO'] = 'baz.bar'
  3332. config_dict = config.load(build_config_details({
  3333. 'version': '3.1',
  3334. 'secrets': {
  3335. 'secretdata': {
  3336. 'external': {'name': '$FOO'}
  3337. }
  3338. }
  3339. }))
  3340. assert config_dict.secrets == {
  3341. 'secretdata': {
  3342. 'external': {'name': 'baz.bar'},
  3343. 'name': 'baz.bar'
  3344. }
  3345. }
  3346. @mock.patch.dict(os.environ)
  3347. def test_interpolation_configs_section(self):
  3348. os.environ['FOO'] = 'baz.bar'
  3349. config_dict = config.load(build_config_details({
  3350. 'version': '3.3',
  3351. 'configs': {
  3352. 'configdata': {
  3353. 'external': {'name': '$FOO'}
  3354. }
  3355. }
  3356. }))
  3357. assert config_dict.configs == {
  3358. 'configdata': {
  3359. 'external': {'name': 'baz.bar'},
  3360. 'name': 'baz.bar'
  3361. }
  3362. }
  3363. class VolumeConfigTest(unittest.TestCase):
  3364. def test_no_binding(self):
  3365. d = make_service_dict('foo', {'build': '.', 'volumes': ['/data']}, working_dir='.')
  3366. assert d['volumes'] == ['/data']
  3367. @mock.patch.dict(os.environ)
  3368. def test_volume_binding_with_environment_variable(self):
  3369. os.environ['VOLUME_PATH'] = '/host/path'
  3370. d = config.load(
  3371. build_config_details(
  3372. {'foo': {'build': '.', 'volumes': ['${VOLUME_PATH}:/container/path']}},
  3373. '.',
  3374. None,
  3375. )
  3376. ).services[0]
  3377. assert d['volumes'] == [VolumeSpec.parse('/host/path:/container/path')]
  3378. @pytest.mark.skipif(IS_WINDOWS_PLATFORM, reason='posix paths')
  3379. def test_volumes_order_is_preserved(self):
  3380. volumes = ['/{0}:/{0}'.format(i) for i in range(0, 6)]
  3381. shuffle(volumes)
  3382. cfg = make_service_dict('foo', {'build': '.', 'volumes': volumes})
  3383. assert cfg['volumes'] == volumes
  3384. @pytest.mark.skipif(IS_WINDOWS_PLATFORM, reason='posix paths')
  3385. @mock.patch.dict(os.environ)
  3386. def test_volume_binding_with_home(self):
  3387. os.environ['HOME'] = '/home/user'
  3388. d = make_service_dict('foo', {'build': '.', 'volumes': ['~:/container/path']}, working_dir='.')
  3389. assert d['volumes'] == ['/home/user:/container/path']
  3390. def test_name_does_not_expand(self):
  3391. d = make_service_dict('foo', {'build': '.', 'volumes': ['mydatavolume:/data']}, working_dir='.')
  3392. assert d['volumes'] == ['mydatavolume:/data']
  3393. def test_absolute_posix_path_does_not_expand(self):
  3394. d = make_service_dict('foo', {'build': '.', 'volumes': ['/var/lib/data:/data']}, working_dir='.')
  3395. assert d['volumes'] == ['/var/lib/data:/data']
  3396. def test_absolute_windows_path_does_not_expand(self):
  3397. d = make_service_dict('foo', {'build': '.', 'volumes': ['c:\\data:/data']}, working_dir='.')
  3398. assert d['volumes'] == ['c:\\data:/data']
  3399. @pytest.mark.skipif(IS_WINDOWS_PLATFORM, reason='posix paths')
  3400. def test_relative_path_does_expand_posix(self):
  3401. d = make_service_dict(
  3402. 'foo',
  3403. {'build': '.', 'volumes': ['./data:/data']},
  3404. working_dir='/home/me/myproject')
  3405. assert d['volumes'] == ['/home/me/myproject/data:/data']
  3406. d = make_service_dict(
  3407. 'foo',
  3408. {'build': '.', 'volumes': ['.:/data']},
  3409. working_dir='/home/me/myproject')
  3410. assert d['volumes'] == ['/home/me/myproject:/data']
  3411. d = make_service_dict(
  3412. 'foo',
  3413. {'build': '.', 'volumes': ['../otherproject:/data']},
  3414. working_dir='/home/me/myproject')
  3415. assert d['volumes'] == ['/home/me/otherproject:/data']
  3416. @pytest.mark.skipif(not IS_WINDOWS_PLATFORM, reason='windows paths')
  3417. def test_relative_path_does_expand_windows(self):
  3418. d = make_service_dict(
  3419. 'foo',
  3420. {'build': '.', 'volumes': ['./data:/data']},
  3421. working_dir='c:\\Users\\me\\myproject')
  3422. assert d['volumes'] == ['c:\\Users\\me\\myproject\\data:/data']
  3423. d = make_service_dict(
  3424. 'foo',
  3425. {'build': '.', 'volumes': ['.:/data']},
  3426. working_dir='c:\\Users\\me\\myproject')
  3427. assert d['volumes'] == ['c:\\Users\\me\\myproject:/data']
  3428. d = make_service_dict(
  3429. 'foo',
  3430. {'build': '.', 'volumes': ['../otherproject:/data']},
  3431. working_dir='c:\\Users\\me\\myproject')
  3432. assert d['volumes'] == ['c:\\Users\\me\\otherproject:/data']
  3433. @mock.patch.dict(os.environ)
  3434. def test_home_directory_with_driver_does_not_expand(self):
  3435. os.environ['NAME'] = 'surprise!'
  3436. d = make_service_dict('foo', {
  3437. 'build': '.',
  3438. 'volumes': ['~:/data'],
  3439. 'volume_driver': 'foodriver',
  3440. }, working_dir='.')
  3441. assert d['volumes'] == ['~:/data']
  3442. def test_volume_path_with_non_ascii_directory(self):
  3443. volume = u'/Füü/data:/data'
  3444. container_path = config.resolve_volume_path(".", volume)
  3445. assert container_path == volume
  3446. class MergePathMappingTest(object):
  3447. config_name = ""
  3448. def test_empty(self):
  3449. service_dict = config.merge_service_dicts({}, {}, DEFAULT_VERSION)
  3450. assert self.config_name not in service_dict
  3451. def test_no_override(self):
  3452. service_dict = config.merge_service_dicts(
  3453. {self.config_name: ['/foo:/code', '/data']},
  3454. {},
  3455. DEFAULT_VERSION)
  3456. assert set(service_dict[self.config_name]) == {'/foo:/code', '/data'}
  3457. def test_no_base(self):
  3458. service_dict = config.merge_service_dicts(
  3459. {},
  3460. {self.config_name: ['/bar:/code']},
  3461. DEFAULT_VERSION)
  3462. assert set(service_dict[self.config_name]) == {'/bar:/code'}
  3463. def test_override_explicit_path(self):
  3464. service_dict = config.merge_service_dicts(
  3465. {self.config_name: ['/foo:/code', '/data']},
  3466. {self.config_name: ['/bar:/code']},
  3467. DEFAULT_VERSION)
  3468. assert set(service_dict[self.config_name]) == {'/bar:/code', '/data'}
  3469. def test_add_explicit_path(self):
  3470. service_dict = config.merge_service_dicts(
  3471. {self.config_name: ['/foo:/code', '/data']},
  3472. {self.config_name: ['/bar:/code', '/quux:/data']},
  3473. DEFAULT_VERSION)
  3474. assert set(service_dict[self.config_name]) == {'/bar:/code', '/quux:/data'}
  3475. def test_remove_explicit_path(self):
  3476. service_dict = config.merge_service_dicts(
  3477. {self.config_name: ['/foo:/code', '/quux:/data']},
  3478. {self.config_name: ['/bar:/code', '/data']},
  3479. DEFAULT_VERSION)
  3480. assert set(service_dict[self.config_name]) == {'/bar:/code', '/data'}
  3481. class MergeVolumesTest(unittest.TestCase, MergePathMappingTest):
  3482. config_name = 'volumes'
  3483. class MergeDevicesTest(unittest.TestCase, MergePathMappingTest):
  3484. config_name = 'devices'
  3485. class BuildOrImageMergeTest(unittest.TestCase):
  3486. def test_merge_build_or_image_no_override(self):
  3487. assert config.merge_service_dicts({'build': '.'}, {}, V1) == {'build': '.'}
  3488. assert config.merge_service_dicts({'image': 'redis'}, {}, V1) == {'image': 'redis'}
  3489. def test_merge_build_or_image_override_with_same(self):
  3490. assert config.merge_service_dicts({'build': '.'}, {'build': './web'}, V1) == {'build': './web'}
  3491. assert config.merge_service_dicts({'image': 'redis'}, {'image': 'postgres'}, V1) == {
  3492. 'image': 'postgres'
  3493. }
  3494. def test_merge_build_or_image_override_with_other(self):
  3495. assert config.merge_service_dicts({'build': '.'}, {'image': 'redis'}, V1) == {
  3496. 'image': 'redis'
  3497. }
  3498. assert config.merge_service_dicts({'image': 'redis'}, {'build': '.'}, V1) == {'build': '.'}
  3499. class MergeListsTest(object):
  3500. config_name = ""
  3501. base_config = []
  3502. override_config = []
  3503. def merged_config(self):
  3504. return set(self.base_config) | set(self.override_config)
  3505. def test_empty(self):
  3506. assert self.config_name not in config.merge_service_dicts({}, {}, DEFAULT_VERSION)
  3507. def test_no_override(self):
  3508. service_dict = config.merge_service_dicts(
  3509. {self.config_name: self.base_config},
  3510. {},
  3511. DEFAULT_VERSION)
  3512. assert set(service_dict[self.config_name]) == set(self.base_config)
  3513. def test_no_base(self):
  3514. service_dict = config.merge_service_dicts(
  3515. {},
  3516. {self.config_name: self.base_config},
  3517. DEFAULT_VERSION)
  3518. assert set(service_dict[self.config_name]) == set(self.base_config)
  3519. def test_add_item(self):
  3520. service_dict = config.merge_service_dicts(
  3521. {self.config_name: self.base_config},
  3522. {self.config_name: self.override_config},
  3523. DEFAULT_VERSION)
  3524. assert set(service_dict[self.config_name]) == set(self.merged_config())
  3525. class MergePortsTest(unittest.TestCase, MergeListsTest):
  3526. config_name = 'ports'
  3527. base_config = ['10:8000', '9000']
  3528. override_config = ['20:8000']
  3529. def merged_config(self):
  3530. return self.convert(self.base_config) | self.convert(self.override_config)
  3531. def convert(self, port_config):
  3532. return set(config.merge_service_dicts(
  3533. {self.config_name: port_config},
  3534. {self.config_name: []},
  3535. DEFAULT_VERSION
  3536. )[self.config_name])
  3537. def test_duplicate_port_mappings(self):
  3538. service_dict = config.merge_service_dicts(
  3539. {self.config_name: self.base_config},
  3540. {self.config_name: self.base_config},
  3541. DEFAULT_VERSION
  3542. )
  3543. assert set(service_dict[self.config_name]) == self.convert(self.base_config)
  3544. def test_no_override(self):
  3545. service_dict = config.merge_service_dicts(
  3546. {self.config_name: self.base_config},
  3547. {},
  3548. DEFAULT_VERSION)
  3549. assert set(service_dict[self.config_name]) == self.convert(self.base_config)
  3550. def test_no_base(self):
  3551. service_dict = config.merge_service_dicts(
  3552. {},
  3553. {self.config_name: self.base_config},
  3554. DEFAULT_VERSION)
  3555. assert set(service_dict[self.config_name]) == self.convert(self.base_config)
  3556. class MergeNetworksTest(unittest.TestCase, MergeListsTest):
  3557. config_name = 'networks'
  3558. base_config = {'default': {'aliases': ['foo.bar', 'foo.baz']}}
  3559. override_config = {'default': {'ipv4_address': '123.234.123.234'}}
  3560. def test_no_network_overrides(self):
  3561. service_dict = config.merge_service_dicts(
  3562. {self.config_name: self.base_config},
  3563. {self.config_name: self.override_config},
  3564. DEFAULT_VERSION)
  3565. assert service_dict[self.config_name] == {
  3566. 'default': {
  3567. 'aliases': ['foo.bar', 'foo.baz'],
  3568. 'ipv4_address': '123.234.123.234'
  3569. }
  3570. }
  3571. def test_network_has_none_value(self):
  3572. service_dict = config.merge_service_dicts(
  3573. {self.config_name: {
  3574. 'default': None
  3575. }},
  3576. {self.config_name: {
  3577. 'default': {
  3578. 'aliases': []
  3579. }
  3580. }},
  3581. DEFAULT_VERSION)
  3582. assert service_dict[self.config_name] == {
  3583. 'default': {
  3584. 'aliases': []
  3585. }
  3586. }
  3587. def test_all_properties(self):
  3588. service_dict = config.merge_service_dicts(
  3589. {self.config_name: {
  3590. 'default': {
  3591. 'aliases': ['foo.bar', 'foo.baz'],
  3592. 'link_local_ips': ['192.168.1.10', '192.168.1.11'],
  3593. 'ipv4_address': '111.111.111.111',
  3594. 'ipv6_address': 'FE80:CD00:0000:0CDE:1257:0000:211E:729C-first'
  3595. }
  3596. }},
  3597. {self.config_name: {
  3598. 'default': {
  3599. 'aliases': ['foo.baz', 'foo.baz2'],
  3600. 'link_local_ips': ['192.168.1.11', '192.168.1.12'],
  3601. 'ipv4_address': '123.234.123.234',
  3602. 'ipv6_address': 'FE80:CD00:0000:0CDE:1257:0000:211E:729C-second'
  3603. }
  3604. }},
  3605. DEFAULT_VERSION)
  3606. assert service_dict[self.config_name] == {
  3607. 'default': {
  3608. 'aliases': ['foo.bar', 'foo.baz', 'foo.baz2'],
  3609. 'link_local_ips': ['192.168.1.10', '192.168.1.11', '192.168.1.12'],
  3610. 'ipv4_address': '123.234.123.234',
  3611. 'ipv6_address': 'FE80:CD00:0000:0CDE:1257:0000:211E:729C-second'
  3612. }
  3613. }
  3614. def test_no_network_name_overrides(self):
  3615. service_dict = config.merge_service_dicts(
  3616. {
  3617. self.config_name: {
  3618. 'default': {
  3619. 'aliases': ['foo.bar', 'foo.baz'],
  3620. 'ipv4_address': '123.234.123.234'
  3621. }
  3622. }
  3623. },
  3624. {
  3625. self.config_name: {
  3626. 'another_network': {
  3627. 'ipv4_address': '123.234.123.234'
  3628. }
  3629. }
  3630. },
  3631. DEFAULT_VERSION)
  3632. assert service_dict[self.config_name] == {
  3633. 'default': {
  3634. 'aliases': ['foo.bar', 'foo.baz'],
  3635. 'ipv4_address': '123.234.123.234'
  3636. },
  3637. 'another_network': {
  3638. 'ipv4_address': '123.234.123.234'
  3639. }
  3640. }
  3641. class MergeStringsOrListsTest(unittest.TestCase):
  3642. def test_no_override(self):
  3643. service_dict = config.merge_service_dicts(
  3644. {'dns': '8.8.8.8'},
  3645. {},
  3646. DEFAULT_VERSION)
  3647. assert set(service_dict['dns']) == {'8.8.8.8'}
  3648. def test_no_base(self):
  3649. service_dict = config.merge_service_dicts(
  3650. {},
  3651. {'dns': '8.8.8.8'},
  3652. DEFAULT_VERSION)
  3653. assert set(service_dict['dns']) == {'8.8.8.8'}
  3654. def test_add_string(self):
  3655. service_dict = config.merge_service_dicts(
  3656. {'dns': ['8.8.8.8']},
  3657. {'dns': '9.9.9.9'},
  3658. DEFAULT_VERSION)
  3659. assert set(service_dict['dns']) == {'8.8.8.8', '9.9.9.9'}
  3660. def test_add_list(self):
  3661. service_dict = config.merge_service_dicts(
  3662. {'dns': '8.8.8.8'},
  3663. {'dns': ['9.9.9.9']},
  3664. DEFAULT_VERSION)
  3665. assert set(service_dict['dns']) == {'8.8.8.8', '9.9.9.9'}
  3666. class MergeLabelsTest(unittest.TestCase):
  3667. def test_empty(self):
  3668. assert 'labels' not in config.merge_service_dicts({}, {}, DEFAULT_VERSION)
  3669. def test_no_override(self):
  3670. service_dict = config.merge_service_dicts(
  3671. make_service_dict('foo', {'build': '.', 'labels': ['foo=1', 'bar']}, 'tests/'),
  3672. make_service_dict('foo', {'build': '.'}, 'tests/'),
  3673. DEFAULT_VERSION)
  3674. assert service_dict['labels'] == {'foo': '1', 'bar': ''}
  3675. def test_no_base(self):
  3676. service_dict = config.merge_service_dicts(
  3677. make_service_dict('foo', {'build': '.'}, 'tests/'),
  3678. make_service_dict('foo', {'build': '.', 'labels': ['foo=2']}, 'tests/'),
  3679. DEFAULT_VERSION)
  3680. assert service_dict['labels'] == {'foo': '2'}
  3681. def test_override_explicit_value(self):
  3682. service_dict = config.merge_service_dicts(
  3683. make_service_dict('foo', {'build': '.', 'labels': ['foo=1', 'bar']}, 'tests/'),
  3684. make_service_dict('foo', {'build': '.', 'labels': ['foo=2']}, 'tests/'),
  3685. DEFAULT_VERSION)
  3686. assert service_dict['labels'] == {'foo': '2', 'bar': ''}
  3687. def test_add_explicit_value(self):
  3688. service_dict = config.merge_service_dicts(
  3689. make_service_dict('foo', {'build': '.', 'labels': ['foo=1', 'bar']}, 'tests/'),
  3690. make_service_dict('foo', {'build': '.', 'labels': ['bar=2']}, 'tests/'),
  3691. DEFAULT_VERSION)
  3692. assert service_dict['labels'] == {'foo': '1', 'bar': '2'}
  3693. def test_remove_explicit_value(self):
  3694. service_dict = config.merge_service_dicts(
  3695. make_service_dict('foo', {'build': '.', 'labels': ['foo=1', 'bar=2']}, 'tests/'),
  3696. make_service_dict('foo', {'build': '.', 'labels': ['bar']}, 'tests/'),
  3697. DEFAULT_VERSION)
  3698. assert service_dict['labels'] == {'foo': '1', 'bar': ''}
  3699. class MergeBuildTest(unittest.TestCase):
  3700. def test_full(self):
  3701. base = {
  3702. 'context': '.',
  3703. 'dockerfile': 'Dockerfile',
  3704. 'args': {
  3705. 'x': '1',
  3706. 'y': '2',
  3707. },
  3708. 'cache_from': ['ubuntu'],
  3709. 'labels': ['com.docker.compose.test=true']
  3710. }
  3711. override = {
  3712. 'context': './prod',
  3713. 'dockerfile': 'Dockerfile.prod',
  3714. 'args': ['x=12'],
  3715. 'cache_from': ['debian'],
  3716. 'labels': {
  3717. 'com.docker.compose.test': 'false',
  3718. 'com.docker.compose.prod': 'true',
  3719. }
  3720. }
  3721. result = config.merge_build(None, {'build': base}, {'build': override})
  3722. assert result['context'] == override['context']
  3723. assert result['dockerfile'] == override['dockerfile']
  3724. assert result['args'] == {'x': '12', 'y': '2'}
  3725. assert set(result['cache_from']) == {'ubuntu', 'debian'}
  3726. assert result['labels'] == override['labels']
  3727. def test_empty_override(self):
  3728. base = {
  3729. 'context': '.',
  3730. 'dockerfile': 'Dockerfile',
  3731. 'args': {
  3732. 'x': '1',
  3733. 'y': '2',
  3734. },
  3735. 'cache_from': ['ubuntu'],
  3736. 'labels': {
  3737. 'com.docker.compose.test': 'true'
  3738. }
  3739. }
  3740. override = {}
  3741. result = config.merge_build(None, {'build': base}, {'build': override})
  3742. assert result == base
  3743. def test_empty_base(self):
  3744. base = {}
  3745. override = {
  3746. 'context': './prod',
  3747. 'dockerfile': 'Dockerfile.prod',
  3748. 'args': {'x': '12'},
  3749. 'cache_from': ['debian'],
  3750. 'labels': {
  3751. 'com.docker.compose.test': 'false',
  3752. 'com.docker.compose.prod': 'true',
  3753. }
  3754. }
  3755. result = config.merge_build(None, {'build': base}, {'build': override})
  3756. assert result == override
  3757. class MemoryOptionsTest(unittest.TestCase):
  3758. def test_validation_fails_with_just_memswap_limit(self):
  3759. """
  3760. When you set a 'memswap_limit' it is invalid config unless you also set
  3761. a mem_limit
  3762. """
  3763. with pytest.raises(ConfigurationError) as excinfo:
  3764. config.load(
  3765. build_config_details(
  3766. {
  3767. 'foo': {'image': 'busybox', 'memswap_limit': 2000000},
  3768. },
  3769. 'tests/fixtures/extends',
  3770. 'filename.yml'
  3771. )
  3772. )
  3773. assert "foo.memswap_limit is invalid: when defining " \
  3774. "'memswap_limit' you must set 'mem_limit' as well" \
  3775. in excinfo.exconly()
  3776. def test_validation_with_correct_memswap_values(self):
  3777. service_dict = config.load(
  3778. build_config_details(
  3779. {'foo': {'image': 'busybox', 'mem_limit': 1000000, 'memswap_limit': 2000000}},
  3780. 'tests/fixtures/extends',
  3781. 'common.yml'
  3782. )
  3783. ).services
  3784. assert service_dict[0]['memswap_limit'] == 2000000
  3785. def test_memswap_can_be_a_string(self):
  3786. service_dict = config.load(
  3787. build_config_details(
  3788. {'foo': {'image': 'busybox', 'mem_limit': "1G", 'memswap_limit': "512M"}},
  3789. 'tests/fixtures/extends',
  3790. 'common.yml'
  3791. )
  3792. ).services
  3793. assert service_dict[0]['memswap_limit'] == "512M"
  3794. class EnvTest(unittest.TestCase):
  3795. def test_parse_environment_as_list(self):
  3796. environment = [
  3797. 'NORMAL=F1',
  3798. 'CONTAINS_EQUALS=F=2',
  3799. 'TRAILING_EQUALS=',
  3800. ]
  3801. assert config.parse_environment(environment) == {
  3802. 'NORMAL': 'F1', 'CONTAINS_EQUALS': 'F=2', 'TRAILING_EQUALS': ''
  3803. }
  3804. def test_parse_environment_as_dict(self):
  3805. environment = {
  3806. 'NORMAL': 'F1',
  3807. 'CONTAINS_EQUALS': 'F=2',
  3808. 'TRAILING_EQUALS': None,
  3809. }
  3810. assert config.parse_environment(environment) == environment
  3811. def test_parse_environment_invalid(self):
  3812. with pytest.raises(ConfigurationError):
  3813. config.parse_environment('a=b')
  3814. def test_parse_environment_empty(self):
  3815. assert config.parse_environment(None) == {}
  3816. @mock.patch.dict(os.environ)
  3817. def test_resolve_environment(self):
  3818. os.environ['FILE_DEF'] = 'E1'
  3819. os.environ['FILE_DEF_EMPTY'] = 'E2'
  3820. os.environ['ENV_DEF'] = 'E3'
  3821. service_dict = {
  3822. 'build': '.',
  3823. 'environment': {
  3824. 'FILE_DEF': 'F1',
  3825. 'FILE_DEF_EMPTY': '',
  3826. 'ENV_DEF': None,
  3827. 'NO_DEF': None
  3828. },
  3829. }
  3830. assert resolve_environment(
  3831. service_dict, Environment.from_env_file(None)
  3832. ) == {'FILE_DEF': 'F1', 'FILE_DEF_EMPTY': '', 'ENV_DEF': 'E3', 'NO_DEF': None}
  3833. def test_resolve_environment_from_env_file(self):
  3834. assert resolve_environment({'env_file': ['tests/fixtures/env/one.env']}) == {
  3835. 'ONE': '2', 'TWO': '1', 'THREE': '3', 'FOO': 'bar'
  3836. }
  3837. def test_environment_overrides_env_file(self):
  3838. assert resolve_environment({
  3839. 'environment': {'FOO': 'baz'},
  3840. 'env_file': ['tests/fixtures/env/one.env'],
  3841. }) == {'ONE': '2', 'TWO': '1', 'THREE': '3', 'FOO': 'baz'}
  3842. def test_resolve_environment_with_multiple_env_files(self):
  3843. service_dict = {
  3844. 'env_file': [
  3845. 'tests/fixtures/env/one.env',
  3846. 'tests/fixtures/env/two.env'
  3847. ]
  3848. }
  3849. assert resolve_environment(service_dict) == {
  3850. 'ONE': '2', 'TWO': '1', 'THREE': '3', 'FOO': 'baz', 'DOO': 'dah'
  3851. }
  3852. def test_resolve_environment_nonexistent_file(self):
  3853. with pytest.raises(ConfigurationError) as exc:
  3854. config.load(build_config_details(
  3855. {'foo': {'image': 'example', 'env_file': 'nonexistent.env'}},
  3856. working_dir='tests/fixtures/env'))
  3857. assert 'Couldn\'t find env file' in exc.exconly()
  3858. assert 'nonexistent.env' in exc.exconly()
  3859. @mock.patch.dict(os.environ)
  3860. def test_resolve_environment_from_env_file_with_empty_values(self):
  3861. os.environ['FILE_DEF'] = 'E1'
  3862. os.environ['FILE_DEF_EMPTY'] = 'E2'
  3863. os.environ['ENV_DEF'] = 'E3'
  3864. assert resolve_environment(
  3865. {'env_file': ['tests/fixtures/env/resolve.env']},
  3866. Environment.from_env_file(None)
  3867. ) == {
  3868. 'FILE_DEF': u'bär',
  3869. 'FILE_DEF_EMPTY': '',
  3870. 'ENV_DEF': 'E3',
  3871. 'NO_DEF': None
  3872. }
  3873. @mock.patch.dict(os.environ)
  3874. def test_resolve_build_args(self):
  3875. os.environ['env_arg'] = 'value2'
  3876. build = {
  3877. 'context': '.',
  3878. 'args': {
  3879. 'arg1': 'value1',
  3880. 'empty_arg': '',
  3881. 'env_arg': None,
  3882. 'no_env': None
  3883. }
  3884. }
  3885. assert resolve_build_args(build['args'], Environment.from_env_file(build['context'])) == {
  3886. 'arg1': 'value1', 'empty_arg': '', 'env_arg': 'value2', 'no_env': None
  3887. }
  3888. @pytest.mark.xfail(IS_WINDOWS_PLATFORM, reason='paths use slash')
  3889. @mock.patch.dict(os.environ)
  3890. def test_resolve_path(self):
  3891. os.environ['HOSTENV'] = '/tmp'
  3892. os.environ['CONTAINERENV'] = '/host/tmp'
  3893. service_dict = config.load(
  3894. build_config_details(
  3895. {'foo': {'build': '.', 'volumes': ['$HOSTENV:$CONTAINERENV']}},
  3896. "tests/fixtures/env",
  3897. )
  3898. ).services[0]
  3899. assert set(service_dict['volumes']) == {VolumeSpec.parse('/tmp:/host/tmp')}
  3900. service_dict = config.load(
  3901. build_config_details(
  3902. {'foo': {'build': '.', 'volumes': ['/opt${HOSTENV}:/opt${CONTAINERENV}']}},
  3903. "tests/fixtures/env",
  3904. )
  3905. ).services[0]
  3906. assert set(service_dict['volumes']) == {VolumeSpec.parse('/opt/tmp:/opt/host/tmp')}
  3907. def load_from_filename(filename, override_dir=None):
  3908. return config.load(
  3909. config.find('.', [filename], Environment.from_env_file('.'), override_dir=override_dir)
  3910. ).services
  3911. class ExtendsTest(unittest.TestCase):
  3912. def test_extends(self):
  3913. service_dicts = load_from_filename('tests/fixtures/extends/docker-compose.yml')
  3914. assert service_sort(service_dicts) == service_sort([
  3915. {
  3916. 'name': 'mydb',
  3917. 'image': 'busybox',
  3918. 'command': 'top',
  3919. },
  3920. {
  3921. 'name': 'myweb',
  3922. 'image': 'busybox',
  3923. 'command': 'top',
  3924. 'network_mode': 'bridge',
  3925. 'links': ['mydb:db'],
  3926. 'environment': {
  3927. "FOO": "1",
  3928. "BAR": "2",
  3929. "BAZ": "2",
  3930. },
  3931. }
  3932. ])
  3933. def test_merging_env_labels_ulimits(self):
  3934. service_dicts = load_from_filename('tests/fixtures/extends/common-env-labels-ulimits.yml')
  3935. assert service_sort(service_dicts) == service_sort([
  3936. {
  3937. 'name': 'web',
  3938. 'image': 'busybox',
  3939. 'command': '/bin/true',
  3940. 'network_mode': 'host',
  3941. 'environment': {
  3942. "FOO": "2",
  3943. "BAR": "1",
  3944. "BAZ": "3",
  3945. },
  3946. 'labels': {'label': 'one'},
  3947. 'ulimits': {'nproc': 65535, 'memlock': {'soft': 1024, 'hard': 2048}}
  3948. }
  3949. ])
  3950. def test_nested(self):
  3951. service_dicts = load_from_filename('tests/fixtures/extends/nested.yml')
  3952. assert service_dicts == [
  3953. {
  3954. 'name': 'myweb',
  3955. 'image': 'busybox',
  3956. 'command': '/bin/true',
  3957. 'network_mode': 'host',
  3958. 'environment': {
  3959. "FOO": "2",
  3960. "BAR": "2",
  3961. },
  3962. },
  3963. ]
  3964. def test_self_referencing_file(self):
  3965. """
  3966. We specify a 'file' key that is the filename we're already in.
  3967. """
  3968. service_dicts = load_from_filename('tests/fixtures/extends/specify-file-as-self.yml')
  3969. assert service_sort(service_dicts) == service_sort([
  3970. {
  3971. 'environment':
  3972. {
  3973. 'YEP': '1', 'BAR': '1', 'BAZ': '3'
  3974. },
  3975. 'image': 'busybox',
  3976. 'name': 'myweb'
  3977. },
  3978. {
  3979. 'environment':
  3980. {'YEP': '1'},
  3981. 'image': 'busybox',
  3982. 'name': 'otherweb'
  3983. },
  3984. {
  3985. 'environment':
  3986. {'YEP': '1', 'BAZ': '3'},
  3987. 'image': 'busybox',
  3988. 'name': 'web'
  3989. }
  3990. ])
  3991. def test_circular(self):
  3992. with pytest.raises(config.CircularReference) as exc:
  3993. load_from_filename('tests/fixtures/extends/circle-1.yml')
  3994. path = [
  3995. (os.path.basename(filename), service_name)
  3996. for (filename, service_name) in exc.value.trail
  3997. ]
  3998. expected = [
  3999. ('circle-1.yml', 'web'),
  4000. ('circle-2.yml', 'other'),
  4001. ('circle-1.yml', 'web'),
  4002. ]
  4003. assert path == expected
  4004. def test_extends_validation_empty_dictionary(self):
  4005. with pytest.raises(ConfigurationError) as excinfo:
  4006. config.load(
  4007. build_config_details(
  4008. {
  4009. 'web': {'image': 'busybox', 'extends': {}},
  4010. },
  4011. 'tests/fixtures/extends',
  4012. 'filename.yml'
  4013. )
  4014. )
  4015. assert 'service' in excinfo.exconly()
  4016. def test_extends_validation_missing_service_key(self):
  4017. with pytest.raises(ConfigurationError) as excinfo:
  4018. config.load(
  4019. build_config_details(
  4020. {
  4021. 'web': {'image': 'busybox', 'extends': {'file': 'common.yml'}},
  4022. },
  4023. 'tests/fixtures/extends',
  4024. 'filename.yml'
  4025. )
  4026. )
  4027. assert "'service' is a required property" in excinfo.exconly()
  4028. def test_extends_validation_invalid_key(self):
  4029. with pytest.raises(ConfigurationError) as excinfo:
  4030. config.load(
  4031. build_config_details(
  4032. {
  4033. 'web': {
  4034. 'image': 'busybox',
  4035. 'extends': {
  4036. 'file': 'common.yml',
  4037. 'service': 'web',
  4038. 'rogue_key': 'is not allowed'
  4039. }
  4040. },
  4041. },
  4042. 'tests/fixtures/extends',
  4043. 'filename.yml'
  4044. )
  4045. )
  4046. assert "web.extends contains unsupported option: 'rogue_key'" \
  4047. in excinfo.exconly()
  4048. def test_extends_validation_sub_property_key(self):
  4049. with pytest.raises(ConfigurationError) as excinfo:
  4050. config.load(
  4051. build_config_details(
  4052. {
  4053. 'web': {
  4054. 'image': 'busybox',
  4055. 'extends': {
  4056. 'file': 1,
  4057. 'service': 'web',
  4058. }
  4059. },
  4060. },
  4061. 'tests/fixtures/extends',
  4062. 'filename.yml'
  4063. )
  4064. )
  4065. assert "web.extends.file contains 1, which is an invalid type, it should be a string" \
  4066. in excinfo.exconly()
  4067. def test_extends_validation_no_file_key_no_filename_set(self):
  4068. dictionary = {'extends': {'service': 'web'}}
  4069. with pytest.raises(ConfigurationError) as excinfo:
  4070. make_service_dict('myweb', dictionary, working_dir='tests/fixtures/extends')
  4071. assert 'file' in excinfo.exconly()
  4072. def test_extends_validation_valid_config(self):
  4073. service = config.load(
  4074. build_config_details(
  4075. {
  4076. 'web': {'image': 'busybox', 'extends': {'service': 'web', 'file': 'common.yml'}},
  4077. },
  4078. 'tests/fixtures/extends',
  4079. 'common.yml'
  4080. )
  4081. ).services
  4082. assert len(service) == 1
  4083. assert isinstance(service[0], dict)
  4084. assert service[0]['command'] == "/bin/true"
  4085. def test_extended_service_with_invalid_config(self):
  4086. with pytest.raises(ConfigurationError) as exc:
  4087. load_from_filename('tests/fixtures/extends/service-with-invalid-schema.yml')
  4088. assert (
  4089. "myweb has neither an image nor a build context specified" in
  4090. exc.exconly()
  4091. )
  4092. def test_extended_service_with_valid_config(self):
  4093. service = load_from_filename('tests/fixtures/extends/service-with-valid-composite-extends.yml')
  4094. assert service[0]['command'] == "top"
  4095. def test_extends_file_defaults_to_self(self):
  4096. """
  4097. Test not specifying a file in our extends options that the
  4098. config is valid and correctly extends from itself.
  4099. """
  4100. service_dicts = load_from_filename('tests/fixtures/extends/no-file-specified.yml')
  4101. assert service_sort(service_dicts) == service_sort([
  4102. {
  4103. 'name': 'myweb',
  4104. 'image': 'busybox',
  4105. 'environment': {
  4106. "BAR": "1",
  4107. "BAZ": "3",
  4108. }
  4109. },
  4110. {
  4111. 'name': 'web',
  4112. 'image': 'busybox',
  4113. 'environment': {
  4114. "BAZ": "3",
  4115. }
  4116. }
  4117. ])
  4118. def test_invalid_links_in_extended_service(self):
  4119. with pytest.raises(ConfigurationError) as excinfo:
  4120. load_from_filename('tests/fixtures/extends/invalid-links.yml')
  4121. assert "services with 'links' cannot be extended" in excinfo.exconly()
  4122. def test_invalid_volumes_from_in_extended_service(self):
  4123. with pytest.raises(ConfigurationError) as excinfo:
  4124. load_from_filename('tests/fixtures/extends/invalid-volumes.yml')
  4125. assert "services with 'volumes_from' cannot be extended" in excinfo.exconly()
  4126. def test_invalid_net_in_extended_service(self):
  4127. with pytest.raises(ConfigurationError) as excinfo:
  4128. load_from_filename('tests/fixtures/extends/invalid-net-v2.yml')
  4129. assert 'network_mode: service' in excinfo.exconly()
  4130. assert 'cannot be extended' in excinfo.exconly()
  4131. with pytest.raises(ConfigurationError) as excinfo:
  4132. load_from_filename('tests/fixtures/extends/invalid-net.yml')
  4133. assert 'net: container' in excinfo.exconly()
  4134. assert 'cannot be extended' in excinfo.exconly()
  4135. @mock.patch.dict(os.environ)
  4136. def test_load_config_runs_interpolation_in_extended_service(self):
  4137. os.environ.update(HOSTNAME_VALUE="penguin")
  4138. expected_interpolated_value = "host-penguin"
  4139. service_dicts = load_from_filename(
  4140. 'tests/fixtures/extends/valid-interpolation.yml')
  4141. for service in service_dicts:
  4142. assert service['hostname'] == expected_interpolated_value
  4143. @pytest.mark.xfail(IS_WINDOWS_PLATFORM, reason='paths use slash')
  4144. def test_volume_path(self):
  4145. dicts = load_from_filename('tests/fixtures/volume-path/docker-compose.yml')
  4146. paths = [
  4147. VolumeSpec(
  4148. os.path.abspath('tests/fixtures/volume-path/common/foo'),
  4149. '/foo',
  4150. 'rw'),
  4151. VolumeSpec(
  4152. os.path.abspath('tests/fixtures/volume-path/bar'),
  4153. '/bar',
  4154. 'rw')
  4155. ]
  4156. assert set(dicts[0]['volumes']) == set(paths)
  4157. def test_parent_build_path_dne(self):
  4158. child = load_from_filename('tests/fixtures/extends/nonexistent-path-child.yml')
  4159. assert child == [
  4160. {
  4161. 'name': 'dnechild',
  4162. 'image': 'busybox',
  4163. 'command': '/bin/true',
  4164. 'environment': {
  4165. "FOO": "1",
  4166. "BAR": "2",
  4167. },
  4168. },
  4169. ]
  4170. def test_load_throws_error_when_base_service_does_not_exist(self):
  4171. with pytest.raises(ConfigurationError) as excinfo:
  4172. load_from_filename('tests/fixtures/extends/nonexistent-service.yml')
  4173. assert "Cannot extend service 'foo'" in excinfo.exconly()
  4174. assert "Service not found" in excinfo.exconly()
  4175. def test_partial_service_config_in_extends_is_still_valid(self):
  4176. dicts = load_from_filename('tests/fixtures/extends/valid-common-config.yml')
  4177. assert dicts[0]['environment'] == {'FOO': '1'}
  4178. def test_extended_service_with_verbose_and_shorthand_way(self):
  4179. services = load_from_filename('tests/fixtures/extends/verbose-and-shorthand.yml')
  4180. assert service_sort(services) == service_sort([
  4181. {
  4182. 'name': 'base',
  4183. 'image': 'busybox',
  4184. 'environment': {'BAR': '1'},
  4185. },
  4186. {
  4187. 'name': 'verbose',
  4188. 'image': 'busybox',
  4189. 'environment': {'BAR': '1', 'FOO': '1'},
  4190. },
  4191. {
  4192. 'name': 'shorthand',
  4193. 'image': 'busybox',
  4194. 'environment': {'BAR': '1', 'FOO': '2'},
  4195. },
  4196. ])
  4197. @mock.patch.dict(os.environ)
  4198. def test_extends_with_environment_and_env_files(self):
  4199. tmpdir = py.test.ensuretemp('test_extends_with_environment')
  4200. self.addCleanup(tmpdir.remove)
  4201. commondir = tmpdir.mkdir('common')
  4202. commondir.join('base.yml').write("""
  4203. app:
  4204. image: 'example/app'
  4205. env_file:
  4206. - 'envs'
  4207. environment:
  4208. - SECRET
  4209. - TEST_ONE=common
  4210. - TEST_TWO=common
  4211. """)
  4212. tmpdir.join('docker-compose.yml').write("""
  4213. ext:
  4214. extends:
  4215. file: common/base.yml
  4216. service: app
  4217. env_file:
  4218. - 'envs'
  4219. environment:
  4220. - THING
  4221. - TEST_ONE=top
  4222. """)
  4223. commondir.join('envs').write("""
  4224. COMMON_ENV_FILE
  4225. TEST_ONE=common-env-file
  4226. TEST_TWO=common-env-file
  4227. TEST_THREE=common-env-file
  4228. TEST_FOUR=common-env-file
  4229. """)
  4230. tmpdir.join('envs').write("""
  4231. TOP_ENV_FILE
  4232. TEST_ONE=top-env-file
  4233. TEST_TWO=top-env-file
  4234. TEST_THREE=top-env-file
  4235. """)
  4236. expected = [
  4237. {
  4238. 'name': 'ext',
  4239. 'image': 'example/app',
  4240. 'environment': {
  4241. 'SECRET': 'secret',
  4242. 'TOP_ENV_FILE': 'secret',
  4243. 'COMMON_ENV_FILE': 'secret',
  4244. 'THING': 'thing',
  4245. 'TEST_ONE': 'top',
  4246. 'TEST_TWO': 'common',
  4247. 'TEST_THREE': 'top-env-file',
  4248. 'TEST_FOUR': 'common-env-file',
  4249. },
  4250. },
  4251. ]
  4252. os.environ['SECRET'] = 'secret'
  4253. os.environ['THING'] = 'thing'
  4254. os.environ['COMMON_ENV_FILE'] = 'secret'
  4255. os.environ['TOP_ENV_FILE'] = 'secret'
  4256. config = load_from_filename(str(tmpdir.join('docker-compose.yml')))
  4257. assert config == expected
  4258. def test_extends_with_mixed_versions_is_error(self):
  4259. tmpdir = py.test.ensuretemp('test_extends_with_mixed_version')
  4260. self.addCleanup(tmpdir.remove)
  4261. tmpdir.join('docker-compose.yml').write("""
  4262. version: "2"
  4263. services:
  4264. web:
  4265. extends:
  4266. file: base.yml
  4267. service: base
  4268. image: busybox
  4269. """)
  4270. tmpdir.join('base.yml').write("""
  4271. base:
  4272. volumes: ['/foo']
  4273. ports: ['3000:3000']
  4274. """)
  4275. with pytest.raises(ConfigurationError) as exc:
  4276. load_from_filename(str(tmpdir.join('docker-compose.yml')))
  4277. assert 'Version mismatch' in exc.exconly()
  4278. def test_extends_with_defined_version_passes(self):
  4279. tmpdir = py.test.ensuretemp('test_extends_with_defined_version')
  4280. self.addCleanup(tmpdir.remove)
  4281. tmpdir.join('docker-compose.yml').write("""
  4282. version: "2"
  4283. services:
  4284. web:
  4285. extends:
  4286. file: base.yml
  4287. service: base
  4288. image: busybox
  4289. """)
  4290. tmpdir.join('base.yml').write("""
  4291. version: "2"
  4292. services:
  4293. base:
  4294. volumes: ['/foo']
  4295. ports: ['3000:3000']
  4296. command: top
  4297. """)
  4298. service = load_from_filename(str(tmpdir.join('docker-compose.yml')))
  4299. assert service[0]['command'] == "top"
  4300. def test_extends_with_depends_on(self):
  4301. tmpdir = py.test.ensuretemp('test_extends_with_depends_on')
  4302. self.addCleanup(tmpdir.remove)
  4303. tmpdir.join('docker-compose.yml').write("""
  4304. version: "2"
  4305. services:
  4306. base:
  4307. image: example
  4308. web:
  4309. extends: base
  4310. image: busybox
  4311. depends_on: ['other']
  4312. other:
  4313. image: example
  4314. """)
  4315. services = load_from_filename(str(tmpdir.join('docker-compose.yml')))
  4316. assert service_sort(services)[2]['depends_on'] == {
  4317. 'other': {'condition': 'service_started'}
  4318. }
  4319. def test_extends_with_healthcheck(self):
  4320. service_dicts = load_from_filename('tests/fixtures/extends/healthcheck-2.yml')
  4321. assert service_sort(service_dicts) == [{
  4322. 'name': 'demo',
  4323. 'image': 'foobar:latest',
  4324. 'healthcheck': {
  4325. 'test': ['CMD', '/health.sh'],
  4326. 'interval': 10000000000,
  4327. 'timeout': 5000000000,
  4328. 'retries': 36,
  4329. }
  4330. }]
  4331. def test_extends_with_ports(self):
  4332. tmpdir = py.test.ensuretemp('test_extends_with_ports')
  4333. self.addCleanup(tmpdir.remove)
  4334. tmpdir.join('docker-compose.yml').write("""
  4335. version: '2'
  4336. services:
  4337. a:
  4338. image: nginx
  4339. ports:
  4340. - 80
  4341. b:
  4342. extends:
  4343. service: a
  4344. """)
  4345. services = load_from_filename(str(tmpdir.join('docker-compose.yml')))
  4346. assert len(services) == 2
  4347. for svc in services:
  4348. assert svc['ports'] == [types.ServicePort('80', None, None, None, None)]
  4349. def test_extends_with_security_opt(self):
  4350. tmpdir = py.test.ensuretemp('test_extends_with_ports')
  4351. self.addCleanup(tmpdir.remove)
  4352. tmpdir.join('docker-compose.yml').write("""
  4353. version: '2'
  4354. services:
  4355. a:
  4356. image: nginx
  4357. security_opt:
  4358. - apparmor:unconfined
  4359. - seccomp:unconfined
  4360. b:
  4361. extends:
  4362. service: a
  4363. """)
  4364. services = load_from_filename(str(tmpdir.join('docker-compose.yml')))
  4365. assert len(services) == 2
  4366. for svc in services:
  4367. assert types.SecurityOpt.parse('apparmor:unconfined') in svc['security_opt']
  4368. assert types.SecurityOpt.parse('seccomp:unconfined') in svc['security_opt']
  4369. @mock.patch.object(ConfigFile, 'from_filename', wraps=ConfigFile.from_filename)
  4370. def test_extends_same_file_optimization(self, from_filename_mock):
  4371. load_from_filename('tests/fixtures/extends/no-file-specified.yml')
  4372. from_filename_mock.assert_called_once()
  4373. @pytest.mark.xfail(IS_WINDOWS_PLATFORM, reason='paths use slash')
  4374. class ExpandPathTest(unittest.TestCase):
  4375. working_dir = '/home/user/somedir'
  4376. def test_expand_path_normal(self):
  4377. result = config.expand_path(self.working_dir, 'myfile')
  4378. assert result == self.working_dir + '/' + 'myfile'
  4379. def test_expand_path_absolute(self):
  4380. abs_path = '/home/user/otherdir/somefile'
  4381. result = config.expand_path(self.working_dir, abs_path)
  4382. assert result == abs_path
  4383. def test_expand_path_with_tilde(self):
  4384. test_path = '~/otherdir/somefile'
  4385. with mock.patch.dict(os.environ):
  4386. os.environ['HOME'] = user_path = '/home/user/'
  4387. result = config.expand_path(self.working_dir, test_path)
  4388. assert result == user_path + 'otherdir/somefile'
  4389. class VolumePathTest(unittest.TestCase):
  4390. def test_split_path_mapping_with_windows_path(self):
  4391. host_path = "c:\\Users\\msamblanet\\Documents\\anvil\\connect\\config"
  4392. windows_volume_path = host_path + ":/opt/connect/config:ro"
  4393. expected_mapping = ("/opt/connect/config", (host_path, 'ro'))
  4394. mapping = config.split_path_mapping(windows_volume_path)
  4395. assert mapping == expected_mapping
  4396. def test_split_path_mapping_with_windows_path_in_container(self):
  4397. host_path = 'c:\\Users\\remilia\\data'
  4398. container_path = 'c:\\scarletdevil\\data'
  4399. expected_mapping = (container_path, (host_path, None))
  4400. mapping = config.split_path_mapping('{0}:{1}'.format(host_path, container_path))
  4401. assert mapping == expected_mapping
  4402. def test_split_path_mapping_with_root_mount(self):
  4403. host_path = '/'
  4404. container_path = '/var/hostroot'
  4405. expected_mapping = (container_path, (host_path, None))
  4406. mapping = config.split_path_mapping('{0}:{1}'.format(host_path, container_path))
  4407. assert mapping == expected_mapping
  4408. @pytest.mark.xfail(IS_WINDOWS_PLATFORM, reason='paths use slash')
  4409. class BuildPathTest(unittest.TestCase):
  4410. def setUp(self):
  4411. self.abs_context_path = os.path.join(os.getcwd(), 'tests/fixtures/build-ctx')
  4412. def test_nonexistent_path(self):
  4413. with pytest.raises(ConfigurationError):
  4414. config.load(
  4415. build_config_details(
  4416. {
  4417. 'foo': {'build': 'nonexistent.path'},
  4418. },
  4419. 'working_dir',
  4420. 'filename.yml'
  4421. )
  4422. )
  4423. def test_relative_path(self):
  4424. relative_build_path = '../build-ctx/'
  4425. service_dict = make_service_dict(
  4426. 'relpath',
  4427. {'build': relative_build_path},
  4428. working_dir='tests/fixtures/build-path'
  4429. )
  4430. assert service_dict['build'] == self.abs_context_path
  4431. def test_absolute_path(self):
  4432. service_dict = make_service_dict(
  4433. 'abspath',
  4434. {'build': self.abs_context_path},
  4435. working_dir='tests/fixtures/build-path'
  4436. )
  4437. assert service_dict['build'] == self.abs_context_path
  4438. def test_from_file(self):
  4439. service_dict = load_from_filename('tests/fixtures/build-path/docker-compose.yml')
  4440. assert service_dict == [{'name': 'foo', 'build': {'context': self.abs_context_path}}]
  4441. def test_from_file_override_dir(self):
  4442. override_dir = os.path.join(os.getcwd(), 'tests/fixtures/')
  4443. service_dict = load_from_filename(
  4444. 'tests/fixtures/build-path-override-dir/docker-compose.yml', override_dir=override_dir)
  4445. assert service_dict == [{'name': 'foo', 'build': {'context': self.abs_context_path}}]
  4446. def test_valid_url_in_build_path(self):
  4447. valid_urls = [
  4448. 'git://github.com/docker/docker',
  4449. '[email protected]:docker/docker.git',
  4450. '[email protected]:atlassianlabs/atlassian-docker.git',
  4451. 'https://github.com/docker/docker.git',
  4452. 'http://github.com/docker/docker.git',
  4453. 'github.com/docker/docker.git',
  4454. ]
  4455. for valid_url in valid_urls:
  4456. service_dict = config.load(build_config_details({
  4457. 'validurl': {'build': valid_url},
  4458. }, '.', None)).services
  4459. assert service_dict[0]['build'] == {'context': valid_url}
  4460. def test_invalid_url_in_build_path(self):
  4461. invalid_urls = [
  4462. 'example.com/bogus',
  4463. 'ftp://example.com/',
  4464. '/path/does/not/exist',
  4465. ]
  4466. for invalid_url in invalid_urls:
  4467. with pytest.raises(ConfigurationError) as exc:
  4468. config.load(build_config_details({
  4469. 'invalidurl': {'build': invalid_url},
  4470. }, '.', None))
  4471. assert 'build path' in exc.exconly()
  4472. class HealthcheckTest(unittest.TestCase):
  4473. def test_healthcheck(self):
  4474. config_dict = config.load(
  4475. build_config_details({
  4476. 'version': '2.3',
  4477. 'services': {
  4478. 'test': {
  4479. 'image': 'busybox',
  4480. 'healthcheck': {
  4481. 'test': ['CMD', 'true'],
  4482. 'interval': '1s',
  4483. 'timeout': '1m',
  4484. 'retries': 3,
  4485. 'start_period': '10s',
  4486. }
  4487. }
  4488. }
  4489. })
  4490. )
  4491. serialized_config = yaml.load(serialize_config(config_dict))
  4492. serialized_service = serialized_config['services']['test']
  4493. assert serialized_service['healthcheck'] == {
  4494. 'test': ['CMD', 'true'],
  4495. 'interval': '1s',
  4496. 'timeout': '1m',
  4497. 'retries': 3,
  4498. 'start_period': '10s'
  4499. }
  4500. def test_disable(self):
  4501. config_dict = config.load(
  4502. build_config_details({
  4503. 'version': '2.3',
  4504. 'services': {
  4505. 'test': {
  4506. 'image': 'busybox',
  4507. 'healthcheck': {
  4508. 'disable': True,
  4509. }
  4510. }
  4511. }
  4512. })
  4513. )
  4514. serialized_config = yaml.load(serialize_config(config_dict))
  4515. serialized_service = serialized_config['services']['test']
  4516. assert serialized_service['healthcheck'] == {
  4517. 'test': ['NONE'],
  4518. }
  4519. def test_disable_with_other_config_is_invalid(self):
  4520. with pytest.raises(ConfigurationError) as excinfo:
  4521. config.load(
  4522. build_config_details({
  4523. 'version': '2.3',
  4524. 'services': {
  4525. 'invalid-healthcheck': {
  4526. 'image': 'busybox',
  4527. 'healthcheck': {
  4528. 'disable': True,
  4529. 'interval': '1s',
  4530. }
  4531. }
  4532. }
  4533. })
  4534. )
  4535. assert 'invalid-healthcheck' in excinfo.exconly()
  4536. assert '"disable: true" cannot be combined with other options' in excinfo.exconly()
  4537. def test_healthcheck_with_invalid_test(self):
  4538. with pytest.raises(ConfigurationError) as excinfo:
  4539. config.load(
  4540. build_config_details({
  4541. 'version': '2.3',
  4542. 'services': {
  4543. 'invalid-healthcheck': {
  4544. 'image': 'busybox',
  4545. 'healthcheck': {
  4546. 'test': ['true'],
  4547. 'interval': '1s',
  4548. 'timeout': '1m',
  4549. 'retries': 3,
  4550. 'start_period': '10s',
  4551. }
  4552. }
  4553. }
  4554. })
  4555. )
  4556. assert 'invalid-healthcheck' in excinfo.exconly()
  4557. assert 'the first item must be either NONE, CMD or CMD-SHELL' in excinfo.exconly()
  4558. class GetDefaultConfigFilesTestCase(unittest.TestCase):
  4559. files = [
  4560. 'docker-compose.yml',
  4561. 'docker-compose.yaml',
  4562. ]
  4563. def test_get_config_path_default_file_in_basedir(self):
  4564. for index, filename in enumerate(self.files):
  4565. assert filename == get_config_filename_for_files(self.files[index:])
  4566. with pytest.raises(config.ComposeFileNotFound):
  4567. get_config_filename_for_files([])
  4568. def test_get_config_path_default_file_in_parent_dir(self):
  4569. """Test with files placed in the subdir"""
  4570. def get_config_in_subdir(files):
  4571. return get_config_filename_for_files(files, subdir=True)
  4572. for index, filename in enumerate(self.files):
  4573. assert filename == get_config_in_subdir(self.files[index:])
  4574. with pytest.raises(config.ComposeFileNotFound):
  4575. get_config_in_subdir([])
  4576. def get_config_filename_for_files(filenames, subdir=None):
  4577. def make_files(dirname, filenames):
  4578. for fname in filenames:
  4579. with open(os.path.join(dirname, fname), 'w') as f:
  4580. f.write('')
  4581. project_dir = tempfile.mkdtemp()
  4582. try:
  4583. make_files(project_dir, filenames)
  4584. if subdir:
  4585. base_dir = tempfile.mkdtemp(dir=project_dir)
  4586. else:
  4587. base_dir = project_dir
  4588. filename, = config.get_default_config_files(base_dir)
  4589. return os.path.basename(filename)
  4590. finally:
  4591. shutil.rmtree(project_dir)
  4592. class SerializeTest(unittest.TestCase):
  4593. def test_denormalize_depends_on_v3(self):
  4594. service_dict = {
  4595. 'image': 'busybox',
  4596. 'command': 'true',
  4597. 'depends_on': {
  4598. 'service2': {'condition': 'service_started'},
  4599. 'service3': {'condition': 'service_started'},
  4600. }
  4601. }
  4602. assert denormalize_service_dict(service_dict, V3_0) == {
  4603. 'image': 'busybox',
  4604. 'command': 'true',
  4605. 'depends_on': ['service2', 'service3']
  4606. }
  4607. def test_denormalize_depends_on_v2_1(self):
  4608. service_dict = {
  4609. 'image': 'busybox',
  4610. 'command': 'true',
  4611. 'depends_on': {
  4612. 'service2': {'condition': 'service_started'},
  4613. 'service3': {'condition': 'service_started'},
  4614. }
  4615. }
  4616. assert denormalize_service_dict(service_dict, V2_1) == service_dict
  4617. def test_serialize_time(self):
  4618. data = {
  4619. 9: '9ns',
  4620. 9000: '9us',
  4621. 9000000: '9ms',
  4622. 90000000: '90ms',
  4623. 900000000: '900ms',
  4624. 999999999: '999999999ns',
  4625. 1000000000: '1s',
  4626. 60000000000: '1m',
  4627. 60000000001: '60000000001ns',
  4628. 9000000000000: '150m',
  4629. 90000000000000: '25h',
  4630. }
  4631. for k, v in data.items():
  4632. assert serialize_ns_time_value(k) == v
  4633. def test_denormalize_healthcheck(self):
  4634. service_dict = {
  4635. 'image': 'test',
  4636. 'healthcheck': {
  4637. 'test': 'exit 1',
  4638. 'interval': '1m40s',
  4639. 'timeout': '30s',
  4640. 'retries': 5,
  4641. 'start_period': '2s90ms'
  4642. }
  4643. }
  4644. processed_service = config.process_service(config.ServiceConfig(
  4645. '.', 'test', 'test', service_dict
  4646. ))
  4647. denormalized_service = denormalize_service_dict(processed_service, V2_3)
  4648. assert denormalized_service['healthcheck']['interval'] == '100s'
  4649. assert denormalized_service['healthcheck']['timeout'] == '30s'
  4650. assert denormalized_service['healthcheck']['start_period'] == '2090ms'
  4651. def test_denormalize_image_has_digest(self):
  4652. service_dict = {
  4653. 'image': 'busybox'
  4654. }
  4655. image_digest = 'busybox@sha256:abcde'
  4656. assert denormalize_service_dict(service_dict, V3_0, image_digest) == {
  4657. 'image': 'busybox@sha256:abcde'
  4658. }
  4659. def test_denormalize_image_no_digest(self):
  4660. service_dict = {
  4661. 'image': 'busybox'
  4662. }
  4663. assert denormalize_service_dict(service_dict, V3_0) == {
  4664. 'image': 'busybox'
  4665. }
  4666. def test_serialize_secrets(self):
  4667. service_dict = {
  4668. 'image': 'example/web',
  4669. 'secrets': [
  4670. {'source': 'one'},
  4671. {
  4672. 'source': 'source',
  4673. 'target': 'target',
  4674. 'uid': '100',
  4675. 'gid': '200',
  4676. 'mode': 0o777,
  4677. }
  4678. ]
  4679. }
  4680. secrets_dict = {
  4681. 'one': {'file': '/one.txt'},
  4682. 'source': {'file': '/source.pem'},
  4683. 'two': {'external': True},
  4684. }
  4685. config_dict = config.load(build_config_details({
  4686. 'version': '3.1',
  4687. 'services': {'web': service_dict},
  4688. 'secrets': secrets_dict
  4689. }))
  4690. serialized_config = yaml.load(serialize_config(config_dict))
  4691. serialized_service = serialized_config['services']['web']
  4692. assert secret_sort(serialized_service['secrets']) == secret_sort(service_dict['secrets'])
  4693. assert 'secrets' in serialized_config
  4694. assert serialized_config['secrets']['two'] == secrets_dict['two']
  4695. def test_serialize_ports(self):
  4696. config_dict = config.Config(version=V2_0, services=[
  4697. {
  4698. 'ports': [types.ServicePort('80', '8080', None, None, None)],
  4699. 'image': 'alpine',
  4700. 'name': 'web'
  4701. }
  4702. ], volumes={}, networks={}, secrets={}, configs={})
  4703. serialized_config = yaml.load(serialize_config(config_dict))
  4704. assert '8080:80/tcp' in serialized_config['services']['web']['ports']
  4705. def test_serialize_ports_with_ext_ip(self):
  4706. config_dict = config.Config(version=V3_5, services=[
  4707. {
  4708. 'ports': [types.ServicePort('80', '8080', None, None, '127.0.0.1')],
  4709. 'image': 'alpine',
  4710. 'name': 'web'
  4711. }
  4712. ], volumes={}, networks={}, secrets={}, configs={})
  4713. serialized_config = yaml.load(serialize_config(config_dict))
  4714. assert '127.0.0.1:8080:80/tcp' in serialized_config['services']['web']['ports']
  4715. def test_serialize_configs(self):
  4716. service_dict = {
  4717. 'image': 'example/web',
  4718. 'configs': [
  4719. {'source': 'one'},
  4720. {
  4721. 'source': 'source',
  4722. 'target': 'target',
  4723. 'uid': '100',
  4724. 'gid': '200',
  4725. 'mode': 0o777,
  4726. }
  4727. ]
  4728. }
  4729. configs_dict = {
  4730. 'one': {'file': '/one.txt'},
  4731. 'source': {'file': '/source.pem'},
  4732. 'two': {'external': True},
  4733. }
  4734. config_dict = config.load(build_config_details({
  4735. 'version': '3.3',
  4736. 'services': {'web': service_dict},
  4737. 'configs': configs_dict
  4738. }))
  4739. serialized_config = yaml.load(serialize_config(config_dict))
  4740. serialized_service = serialized_config['services']['web']
  4741. assert secret_sort(serialized_service['configs']) == secret_sort(service_dict['configs'])
  4742. assert 'configs' in serialized_config
  4743. assert serialized_config['configs']['two'] == configs_dict['two']
  4744. def test_serialize_bool_string(self):
  4745. cfg = {
  4746. 'version': '2.2',
  4747. 'services': {
  4748. 'web': {
  4749. 'image': 'example/web',
  4750. 'command': 'true',
  4751. 'environment': {'FOO': 'Y', 'BAR': 'on'}
  4752. }
  4753. }
  4754. }
  4755. config_dict = config.load(build_config_details(cfg))
  4756. serialized_config = serialize_config(config_dict)
  4757. assert 'command: "true"\n' in serialized_config
  4758. assert 'FOO: "Y"\n' in serialized_config
  4759. assert 'BAR: "on"\n' in serialized_config
  4760. def test_serialize_escape_dollar_sign(self):
  4761. cfg = {
  4762. 'version': '2.2',
  4763. 'services': {
  4764. 'web': {
  4765. 'image': 'busybox',
  4766. 'command': 'echo $$FOO',
  4767. 'environment': {
  4768. 'CURRENCY': '$$'
  4769. },
  4770. 'entrypoint': ['$$SHELL', '-c'],
  4771. }
  4772. }
  4773. }
  4774. config_dict = config.load(build_config_details(cfg))
  4775. serialized_config = yaml.load(serialize_config(config_dict))
  4776. serialized_service = serialized_config['services']['web']
  4777. assert serialized_service['environment']['CURRENCY'] == '$$'
  4778. assert serialized_service['command'] == 'echo $$FOO'
  4779. assert serialized_service['entrypoint'][0] == '$$SHELL'
  4780. def test_serialize_escape_dont_interpolate(self):
  4781. cfg = {
  4782. 'version': '2.2',
  4783. 'services': {
  4784. 'web': {
  4785. 'image': 'busybox',
  4786. 'command': 'echo $FOO',
  4787. 'environment': {
  4788. 'CURRENCY': '$'
  4789. },
  4790. 'entrypoint': ['$SHELL', '-c'],
  4791. }
  4792. }
  4793. }
  4794. config_dict = config.load(build_config_details(cfg), interpolate=False)
  4795. serialized_config = yaml.load(serialize_config(config_dict, escape_dollar=False))
  4796. serialized_service = serialized_config['services']['web']
  4797. assert serialized_service['environment']['CURRENCY'] == '$'
  4798. assert serialized_service['command'] == 'echo $FOO'
  4799. assert serialized_service['entrypoint'][0] == '$SHELL'
  4800. def test_serialize_unicode_values(self):
  4801. cfg = {
  4802. 'version': '2.3',
  4803. 'services': {
  4804. 'web': {
  4805. 'image': 'busybox',
  4806. 'command': 'echo 十六夜 咲夜'
  4807. }
  4808. }
  4809. }
  4810. config_dict = config.load(build_config_details(cfg))
  4811. serialized_config = yaml.load(serialize_config(config_dict))
  4812. serialized_service = serialized_config['services']['web']
  4813. assert serialized_service['command'] == 'echo 十六夜 咲夜'
  4814. def test_serialize_external_false(self):
  4815. cfg = {
  4816. 'version': '3.4',
  4817. 'volumes': {
  4818. 'test': {
  4819. 'name': 'test-false',
  4820. 'external': False
  4821. }
  4822. }
  4823. }
  4824. config_dict = config.load(build_config_details(cfg))
  4825. serialized_config = yaml.load(serialize_config(config_dict))
  4826. serialized_volume = serialized_config['volumes']['test']
  4827. assert serialized_volume['external'] is False