add azure_rm_webapp (#40005)
* add azure_rm_webapp * fix lint error * resolve comments * fix lint * fix lint * fix lint * fix lint * fix test failure * fix test * fix test * fix test * refine ux * resolve comments * fix lint and test * fix lint * code cleanup * combine windows_frameworks/linux_framework to one * fix lint in doc * fix l int * fix lint * fix bug * fix bug * fix bug * fix bug * fix syntax error * fix lint error * fix bug of wrong indent * fix test issue * update java framework parameter * fix lint error * resolve comments * fix lint * resolve comments * fix bug * update test * fix lint * fix test
This commit is contained in:
parent
d8ba8c03f3
commit
97942cf049
5 changed files with 1186 additions and 2 deletions
|
@ -66,7 +66,8 @@ AZURE_API_PROFILES = {
|
|||
),
|
||||
'NetworkManagementClient': '2017-11-01',
|
||||
'ResourceManagementClient': '2017-05-10',
|
||||
'StorageManagementClient': '2017-10-01'
|
||||
'StorageManagementClient': '2017-10-01',
|
||||
'WebsiteManagementClient': '2016-08-01'
|
||||
},
|
||||
|
||||
'2017-03-09-profile': {
|
||||
|
@ -1046,7 +1047,8 @@ class AzureRMModuleBase(object):
|
|||
self.log('Getting web client')
|
||||
if not self._web_client:
|
||||
self._web_client = self.get_mgmt_svc_client(WebSiteManagementClient,
|
||||
base_url=self._cloud_environment.endpoints.resource_manager)
|
||||
base_url=self._cloud_environment.endpoints.resource_manager,
|
||||
api_version='2016-08-01')
|
||||
return self._web_client
|
||||
|
||||
@property
|
||||
|
|
942
lib/ansible/modules/cloud/azure/azure_rm_webapp.py
Normal file
942
lib/ansible/modules/cloud/azure/azure_rm_webapp.py
Normal file
|
@ -0,0 +1,942 @@
|
|||
#!/usr/bin/python
|
||||
#
|
||||
# Copyright (c) 2018 Yunge Zhu, <yungez@microsoft.com>
|
||||
#
|
||||
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||
|
||||
from __future__ import absolute_import, division, print_function
|
||||
__metaclass__ = type
|
||||
|
||||
|
||||
ANSIBLE_METADATA = {'metadata_version': '1.1',
|
||||
'status': ['preview'],
|
||||
'supported_by': 'community'}
|
||||
|
||||
|
||||
DOCUMENTATION = '''
|
||||
---
|
||||
module: azure_rm_webapp
|
||||
version_added: "2.7"
|
||||
short_description: Manage Web App instance.
|
||||
description:
|
||||
- Create, update and delete instance of Web App.
|
||||
|
||||
options:
|
||||
resource_group:
|
||||
description:
|
||||
- Name of the resource group to which the resource belongs.
|
||||
required: True
|
||||
name:
|
||||
description:
|
||||
- Unique name of the app to create or update. To create or update a deployment slot, use the {slot} parameter.
|
||||
required: True
|
||||
|
||||
location:
|
||||
description:
|
||||
- Resource location. If not set, location from the resource group will be used as default.
|
||||
|
||||
plan:
|
||||
description:
|
||||
- App service plan. Required for creation.
|
||||
- It can be name of existing app service plan in same resource group as web app.
|
||||
- "It can be resource id of existing app service plan. eg.,
|
||||
/subscriptions/<subs_id>/resourceGroups/<resource_group>/providers/Microsoft.Web/serverFarms/<plan_name>"
|
||||
- It can be a dict which contains C(name), C(resource_group), C(sku), C(is_linux) and C(number_of_workers).
|
||||
- C(name). Name of app service plan.
|
||||
- C(resource_group). Resource group name of app service plan.
|
||||
- C(sku). SKU of app service plan. For allowed sku, please refer to U(https://azure.microsoft.com/en-us/pricing/details/app-service/linux/).
|
||||
- C(is_linux). Indicate is linux app service plan. type bool. default False.
|
||||
- C(number_of_workers). Number of workers.
|
||||
|
||||
frameworks:
|
||||
description:
|
||||
- Set of run time framework settings. Each setting is a dictionary.
|
||||
- See U(https://docs.microsoft.com/en-us/azure/app-service/app-service-web-overview) for more info.
|
||||
suboptions:
|
||||
name:
|
||||
description:
|
||||
- Name of the framework.
|
||||
- Supported framework list for Windows web app and Linux web app is different.
|
||||
- For Windows web app, supported names(June 2018) java, net_framework, php, python, node. Multiple framework can be set at same time.
|
||||
- For Linux web app, supported names(June 2018) java, ruby, php, dotnetcore, node. Only one framework can be set.
|
||||
- Java framework is mutually exclusive with others.
|
||||
choices:
|
||||
- java
|
||||
- net_framework
|
||||
- php
|
||||
- python
|
||||
- ruby
|
||||
- dotnetcore
|
||||
- node
|
||||
version:
|
||||
description:
|
||||
- Version of the framework. For linux web app supported value, see U(https://aka.ms/linux-stacks) for more info.
|
||||
- net_framework supported value sample, 'v4.0' for .NET 4.6 and 'v3.0' for .NET 3.5.
|
||||
- php supported value sample, 5.5, 5.6, 7.0.
|
||||
- python supported value sample, e.g., 5.5, 5.6, 7.0.
|
||||
- node supported value sample, 6.6, 6.9.
|
||||
- dotnetcore supported value sample, 1.0, 1,1, 1.2.
|
||||
- ruby supported value sample, 2.3.
|
||||
settings:
|
||||
description:
|
||||
- List of settings of the framework.
|
||||
suboptions:
|
||||
java_container:
|
||||
description: Name of java contaner. This is supported by specific framework C(java) only. eg. Tomcat, Jetty.
|
||||
java_container_version:
|
||||
description:
|
||||
- Versoin of java container. This is supported by specific framework C(java) only.
|
||||
- For Tomcat, eg. 8.0, 8.5, 9.0. For Jetty, eg. 9.1, 9.3.
|
||||
|
||||
container_settings:
|
||||
description: Web app container settings.
|
||||
suboptions:
|
||||
name:
|
||||
description: Name of container. eg. "imagename:tag"
|
||||
registry_server_url:
|
||||
description: Container registry server url. eg. mydockerregistry.io
|
||||
registry_server_user:
|
||||
description: The container registry server user name.
|
||||
registry_server_password:
|
||||
description:
|
||||
- The container registry server password.
|
||||
|
||||
scm_type:
|
||||
description:
|
||||
- Repository type of deployment source. Eg. LocalGit, GitHub.
|
||||
- Please see U(https://docs.microsoft.com/en-us/rest/api/appservice/webapps/createorupdate#scmtype) for more info.
|
||||
|
||||
deployment_source:
|
||||
description:
|
||||
- Deployment source for git
|
||||
suboptions:
|
||||
url:
|
||||
description:
|
||||
- Repository url of deployment source.
|
||||
|
||||
branch:
|
||||
description:
|
||||
- The branch name of the repository.
|
||||
startup_file:
|
||||
description:
|
||||
- The web's startup file.
|
||||
- This only applies for linux web app.
|
||||
|
||||
client_affinity_enabled:
|
||||
description:
|
||||
- "True to enable client affinity; False to stop sending session affinity cookies, which route client requests in the
|
||||
same session to the same instance."
|
||||
type: bool
|
||||
default: True
|
||||
|
||||
https_only:
|
||||
description:
|
||||
- Configures web site to accept only https requests.
|
||||
type: bool
|
||||
|
||||
dns_registration:
|
||||
description:
|
||||
- If true web app hostname is not registered with DNS on creation.
|
||||
type: bool
|
||||
|
||||
skip_custom_domain_verification:
|
||||
description:
|
||||
- If true, custom (non *.azurewebsites.net) domains associated with web app are not verified.
|
||||
type: bool
|
||||
|
||||
ttl_in_seconds:
|
||||
description:
|
||||
- Time to live in seconds for web app default domain name.
|
||||
|
||||
app_settings:
|
||||
description:
|
||||
- Configure web app application settings. Suboptions are in key value pair format.
|
||||
|
||||
purge_app_settings:
|
||||
description:
|
||||
- Purge any existing application settings. Replace web app application settings with app_settings.
|
||||
type: bool
|
||||
|
||||
state:
|
||||
description:
|
||||
- Assert the state of the Web App.
|
||||
- Use 'present' to create or update an Web App and 'absent' to delete it.
|
||||
default: present
|
||||
choices:
|
||||
- absent
|
||||
- present
|
||||
|
||||
extends_documentation_fragment:
|
||||
- azure
|
||||
- azure_tags
|
||||
|
||||
author:
|
||||
- "Yunge Zhu(@yungezz)"
|
||||
|
||||
'''
|
||||
|
||||
EXAMPLES = '''
|
||||
- name: Create a windows web app with non-exist app service plan
|
||||
azure_rm_webapp:
|
||||
resource_group: myresourcegroup
|
||||
name: mywinwebapp
|
||||
plan:
|
||||
resource_group: myappserviceplan_rg
|
||||
name: myappserviceplan
|
||||
is_linux: false
|
||||
sku: S1
|
||||
|
||||
- name: Create a docker web app with some app settings, with docker image
|
||||
azure_rm_webapp:
|
||||
resource_group: myresourcegroup
|
||||
name: mydockerwebapp
|
||||
plan:
|
||||
resource_group: appserviceplan_test
|
||||
name: myappplan
|
||||
is_linux: true
|
||||
sku: S1
|
||||
number_of_workers: 2
|
||||
app_settings:
|
||||
testkey: testvalue
|
||||
testkey2: testvalue2
|
||||
container_settings:
|
||||
name: ansible/ansible:ubuntu1404
|
||||
|
||||
- name: Create a docker web app with private acr registry
|
||||
azure_rm_webapp:
|
||||
resource_group: myresourcegroup
|
||||
name: mydockerwebapp
|
||||
plan: myappplan
|
||||
app_settings:
|
||||
testkey: testvalue
|
||||
container_settings:
|
||||
name: ansible/ubuntu1404
|
||||
registry_server_url: myregistry.io
|
||||
registry_server_user: user
|
||||
registry_server_password: pass
|
||||
|
||||
- name: Create a linux web app with Node 6.6 framework
|
||||
azure_rm_webapp:
|
||||
resource_group: myresourcegroup
|
||||
name: mylinuxwebapp
|
||||
plan:
|
||||
resource_group: appserviceplan_test
|
||||
name: myappplan
|
||||
app_settings:
|
||||
testkey: testvalue
|
||||
frameworks:
|
||||
- name: "node"
|
||||
version: "6.6"
|
||||
|
||||
- name: Create a windows web app with node, php
|
||||
azure_rm_webapp:
|
||||
resource_group: myresourcegroup
|
||||
name: mywinwebapp
|
||||
plan:
|
||||
resource_group: appserviceplan_test
|
||||
name: myappplan
|
||||
app_settings:
|
||||
testkey: testvalue
|
||||
frameworks:
|
||||
- name: "node"
|
||||
version: 6.6
|
||||
- name: "php"
|
||||
version: "7.0"
|
||||
|
||||
- name: Create a linux web app with java framework
|
||||
azure_rm_webapp:
|
||||
resource_group: myresourcegroup
|
||||
name: mylinuxwebapp
|
||||
plan:
|
||||
resource_group: appserviceplan_test
|
||||
name: myappplan
|
||||
app_settings:
|
||||
testkey: testvalue
|
||||
frameworks:
|
||||
- name: "java"
|
||||
version: "1.8"
|
||||
settings:
|
||||
java_container: "Tomcat"
|
||||
java_container_version: "8.0"
|
||||
'''
|
||||
|
||||
RETURN = '''
|
||||
azure_webapp:
|
||||
description: Id of current web app.
|
||||
returned: always
|
||||
type: dict
|
||||
sample: {
|
||||
"id": "/subscriptions/<subscription_id>/resourceGroups/ansiblewebapp1/providers/Microsoft.Web/sites/ansiblewindowsaaa"
|
||||
}
|
||||
'''
|
||||
|
||||
import time
|
||||
from ansible.module_utils.azure_rm_common import AzureRMModuleBase
|
||||
|
||||
try:
|
||||
from msrestazure.azure_exceptions import CloudError
|
||||
from msrestazure.azure_operation import AzureOperationPoller
|
||||
from msrest.serialization import Model
|
||||
from azure.mgmt.web.models import (
|
||||
site_config, app_service_plan, Site,
|
||||
AppServicePlan, SkuDescription, NameValuePair
|
||||
)
|
||||
except ImportError:
|
||||
# This is handled in azure_rm_common
|
||||
pass
|
||||
|
||||
container_settings_spec = dict(
|
||||
name=dict(type='str', required=True),
|
||||
registry_server_url=dict(type='str'),
|
||||
registry_server_user=dict(type='str'),
|
||||
registry_server_password=dict(type='str', no_log=True)
|
||||
)
|
||||
|
||||
deployment_source_spec = dict(
|
||||
url=dict(type='str'),
|
||||
branch=dict(type='str')
|
||||
)
|
||||
|
||||
|
||||
framework_settings_spec = dict(
|
||||
java_container=dict(type='str'),
|
||||
java_container_version=dict(type='str')
|
||||
)
|
||||
|
||||
|
||||
framework_spec = dict(
|
||||
name=dict(
|
||||
type='str',
|
||||
required=True,
|
||||
choices=['net_framework', 'java', 'php', 'node', 'python', 'dotnetcore', 'ruby']),
|
||||
version=dict(type='str', required=True),
|
||||
settings=dict(type='dict', options=framework_settings_spec)
|
||||
)
|
||||
|
||||
|
||||
def _normalize_sku(sku):
|
||||
if sku is None:
|
||||
return sku
|
||||
|
||||
sku = sku.upper()
|
||||
if sku == 'FREE':
|
||||
return 'F1'
|
||||
elif sku == 'SHARED':
|
||||
return 'D1'
|
||||
return sku
|
||||
|
||||
|
||||
def get_sku_name(tier):
|
||||
tier = tier.upper()
|
||||
if tier == 'F1' or tier == "FREE":
|
||||
return 'FREE'
|
||||
elif tier == 'D1' or tier == "SHARED":
|
||||
return 'SHARED'
|
||||
elif tier in ['B1', 'B2', 'B3', 'BASIC']:
|
||||
return 'BASIC'
|
||||
elif tier in ['S1', 'S2', 'S3']:
|
||||
return 'STANDARD'
|
||||
elif tier in ['P1', 'P2', 'P3']:
|
||||
return 'PREMIUM'
|
||||
elif tier in ['P1V2', 'P2V2', 'P3V2']:
|
||||
return 'PREMIUMV2'
|
||||
else:
|
||||
return None
|
||||
|
||||
|
||||
class Actions:
|
||||
NoAction, CreateOrUpdate, UpdateAppSettings, Delete = range(4)
|
||||
|
||||
|
||||
class AzureRMWebApps(AzureRMModuleBase):
|
||||
"""Configuration class for an Azure RM Web App resource"""
|
||||
|
||||
def __init__(self):
|
||||
self.module_arg_spec = dict(
|
||||
resource_group=dict(
|
||||
type='str',
|
||||
required=True
|
||||
),
|
||||
name=dict(
|
||||
type='str',
|
||||
required=True
|
||||
),
|
||||
location=dict(
|
||||
type='str'
|
||||
),
|
||||
plan=dict(
|
||||
type='raw'
|
||||
),
|
||||
frameworks=dict(
|
||||
type='list',
|
||||
elements='dict',
|
||||
options=framework_spec
|
||||
),
|
||||
container_settings=dict(
|
||||
type='dict',
|
||||
options=container_settings_spec
|
||||
),
|
||||
scm_type=dict(
|
||||
type='str',
|
||||
),
|
||||
deployment_source=dict(
|
||||
type='dict',
|
||||
options=deployment_source_spec
|
||||
),
|
||||
startup_file=dict(
|
||||
type='str'
|
||||
),
|
||||
client_affinity_enabled=dict(
|
||||
type='bool',
|
||||
default=True
|
||||
),
|
||||
dns_registration=dict(
|
||||
type='bool'
|
||||
),
|
||||
https_only=dict(
|
||||
type='bool'
|
||||
),
|
||||
skip_custom_domain_verification=dict(
|
||||
type='bool'
|
||||
),
|
||||
ttl_in_seconds=dict(
|
||||
type='int'
|
||||
),
|
||||
app_settings=dict(
|
||||
type='dict'
|
||||
),
|
||||
purge_app_settings=dict(
|
||||
type='bool',
|
||||
default=False
|
||||
),
|
||||
state=dict(
|
||||
type='str',
|
||||
default='present',
|
||||
choices=['present', 'absent']
|
||||
)
|
||||
)
|
||||
|
||||
mutually_exclusive = [['container_settings', 'frameworks']]
|
||||
|
||||
self.resource_group = None
|
||||
self.name = None
|
||||
self.location = None
|
||||
|
||||
# update in create_or_update as parameters
|
||||
self.client_affinity_enabled = True
|
||||
self.dns_registration = None
|
||||
self.skip_custom_domain_verification = None
|
||||
self.ttl_in_seconds = None
|
||||
self.https_only = None
|
||||
|
||||
self.tags = None
|
||||
|
||||
# site config, e.g app settings, ssl
|
||||
self.site_config = dict()
|
||||
self.app_settings = dict()
|
||||
self.app_settings_strDic = None
|
||||
|
||||
# app service plan
|
||||
self.plan = None
|
||||
|
||||
# siteSourceControl
|
||||
self.deployment_source = dict()
|
||||
|
||||
# site, used at level creation, or update. e.g windows/linux, client_affinity etc first level args
|
||||
self.site = None
|
||||
|
||||
# property for internal usage, not used for sdk
|
||||
self.container_settings = None
|
||||
|
||||
self.purge_app_settings = False
|
||||
|
||||
self.results = dict(
|
||||
changed=False,
|
||||
id=None,
|
||||
)
|
||||
self.state = None
|
||||
self.to_do = Actions.NoAction
|
||||
|
||||
self.frameworks = None
|
||||
|
||||
# set site_config value from kwargs
|
||||
self.site_config_updatable_properties = ["net_framework_version",
|
||||
"java_version",
|
||||
"php_version",
|
||||
"python_version",
|
||||
"scm_type"]
|
||||
|
||||
# updatable_properties
|
||||
self.updatable_properties = ["client_affinity_enabled",
|
||||
"force_dns_registration",
|
||||
"https_only",
|
||||
"skip_custom_domain_verification",
|
||||
"ttl_in_seconds"]
|
||||
|
||||
self.supported_linux_frameworks = ['ruby', 'php', 'dotnetcore', 'node', 'java']
|
||||
self.supported_windows_frameworks = ['net_framework', 'php', 'python', 'node', 'java']
|
||||
|
||||
super(AzureRMWebApps, self).__init__(derived_arg_spec=self.module_arg_spec,
|
||||
mutually_exclusive=mutually_exclusive,
|
||||
supports_check_mode=True,
|
||||
supports_tags=True)
|
||||
|
||||
def exec_module(self, **kwargs):
|
||||
"""Main module execution method"""
|
||||
|
||||
for key in list(self.module_arg_spec.keys()) + ['tags']:
|
||||
if hasattr(self, key):
|
||||
setattr(self, key, kwargs[key])
|
||||
elif kwargs[key] is not None:
|
||||
if key == "scm_type":
|
||||
self.site_config[key] = kwargs[key]
|
||||
|
||||
old_response = None
|
||||
response = None
|
||||
to_be_updated = False
|
||||
|
||||
# set location
|
||||
resource_group = self.get_resource_group(self.resource_group)
|
||||
if not self.location:
|
||||
self.location = resource_group.location
|
||||
|
||||
# get existing web app
|
||||
old_response = self.get_webapp()
|
||||
|
||||
if old_response:
|
||||
self.results['id'] = old_response['id']
|
||||
|
||||
if self.state == 'present':
|
||||
if not self.plan and not old_response:
|
||||
self.fail("Please specify plan for newly created web app.")
|
||||
|
||||
if not self.plan:
|
||||
self.plan = old_response['server_farm_id']
|
||||
|
||||
self.plan = self.parse_resource_to_dict(self.plan)
|
||||
|
||||
# get app service plan
|
||||
is_linux = False
|
||||
old_plan = self.get_app_service_plan()
|
||||
if old_plan:
|
||||
is_linux = old_plan['reserved']
|
||||
else:
|
||||
is_linux = self.plan['is_linux'] if 'is_linux' in self.plan else False
|
||||
|
||||
if self.frameworks:
|
||||
# java is mutually exclusive with other frameworks
|
||||
if len(self.frameworks) > 1 and any(f['name'] == 'java' for f in self.frameworks):
|
||||
self.fail('Java is mutually exclusive with other frameworks.')
|
||||
|
||||
if is_linux:
|
||||
if len(self.frameworks) != 1:
|
||||
self.fail('Can specify one framework only for Linux web app.')
|
||||
|
||||
if self.frameworks[0]['name'] not in self.supported_linux_frameworks:
|
||||
self.fail('Unsupported framework {0} for Linux web app.'.format(self.frameworks[0]['name']))
|
||||
|
||||
self.site_config['linux_fx_version'] = (self.frameworks[0]['name'] + '|' + self.frameworks[0]['version']).upper()
|
||||
else:
|
||||
for fx in self.frameworks:
|
||||
if fx.get('name') not in self.supported_windows_frameworks:
|
||||
self.fail('Unsupported framework {0} for Windows web app.'.format(fx.get('name')))
|
||||
else:
|
||||
self.site_config[fx.get('name') + '_version'] = fx.get('version')
|
||||
|
||||
for fx in self.frameworks:
|
||||
if 'settings' in fx and fx['settings'] is not None:
|
||||
for key, value in fx['settings'].items():
|
||||
self.site_config[key] = value
|
||||
|
||||
if not self.app_settings:
|
||||
self.app_settings = dict()
|
||||
|
||||
if self.container_settings:
|
||||
linux_fx_version = 'DOCKER|'
|
||||
|
||||
if self.container_settings.get('registry_server_url'):
|
||||
self.app_settings['DOCKER_REGISTRY_SERVER_URL'] = 'https://' + self.container_settings['registry_server_url']
|
||||
|
||||
linux_fx_version += self.container_settings['registry_server_url'] + '/'
|
||||
|
||||
linux_fx_version += self.container_settings['name']
|
||||
|
||||
self.site_config['linux_fx_version'] = linux_fx_version
|
||||
|
||||
if self.container_settings.get('registry_server_user'):
|
||||
self.app_settings['DOCKER_REGISTRY_SERVER_USERNAME'] = self.container_settings['registry_server_user']
|
||||
|
||||
if self.container_settings.get('registry_server_password'):
|
||||
self.app_settings['DOCKER_REGISTRY_SERVER_PASSWORD'] = self.container_settings['registry_server_password']
|
||||
|
||||
# init site
|
||||
self.site = Site(location=self.location, site_config=self.site_config)
|
||||
|
||||
if self.https_only is not None:
|
||||
self.site.https_only = self.https_only
|
||||
|
||||
if self.client_affinity_enabled:
|
||||
self.site.client_affinity_enabled = self.client_affinity_enabled
|
||||
|
||||
# check if the web app already present in the resource group
|
||||
if not old_response:
|
||||
self.log("Web App instance doesn't exist")
|
||||
|
||||
to_be_updated = True
|
||||
self.to_do = Actions.CreateOrUpdate
|
||||
|
||||
# service plan is required for creation
|
||||
if not self.plan:
|
||||
self.fail("Please specify app service plan in plan parameter.")
|
||||
|
||||
if not old_plan:
|
||||
# no existing service plan, create one
|
||||
if (not self.plan.get('name') or not self.plan.get('sku')):
|
||||
self.fail('Please specify name, is_linux, sku in plan')
|
||||
|
||||
if 'location' not in self.plan:
|
||||
plan_resource_group = self.get_resource_group(self.plan['resource_group'])
|
||||
self.plan['location'] = plan_resource_group.location
|
||||
|
||||
old_plan = self.create_app_service_plan()
|
||||
|
||||
self.site.server_farm_id = old_plan['id']
|
||||
|
||||
# if linux, setup startup_file
|
||||
if old_plan.get('is_linux'):
|
||||
if self.startup_file:
|
||||
self.site_config['app_command_line'] = self.startup_file
|
||||
|
||||
# set app setting
|
||||
if self.app_settings:
|
||||
app_settings = []
|
||||
for key in self.app_settings.keys():
|
||||
app_settings.append(NameValuePair(key, self.app_settings[key]))
|
||||
|
||||
self.site_config['app_settings'] = app_settings
|
||||
else:
|
||||
# existing web app, do update
|
||||
self.log("Web App instance already exists")
|
||||
|
||||
self.log('Result: {0}'.format(old_response))
|
||||
|
||||
update_tags, old_response['tags'] = self.update_tags(old_response.get('tags', dict()))
|
||||
|
||||
if update_tags:
|
||||
to_be_updated = True
|
||||
|
||||
# check if root level property changed
|
||||
if self.is_updatable_property_changed(old_response):
|
||||
to_be_updated = True
|
||||
self.to_do = Actions.CreateOrUpdate
|
||||
|
||||
# check if site_config changed
|
||||
old_config = self.get_webapp_configuration()
|
||||
|
||||
if self.is_site_config_changed(old_config):
|
||||
to_be_updated = True
|
||||
self.to_do = Actions.CreateOrUpdate
|
||||
|
||||
# check if linux_fx_version changed
|
||||
if old_config.linux_fx_version != self.site_config.get('linux_fx_version', ''):
|
||||
to_be_updated = True
|
||||
self.to_do = Actions.CreateOrUpdate
|
||||
|
||||
self.app_settings_strDic = self.list_app_settings()
|
||||
|
||||
# purge existing app_settings:
|
||||
if self.purge_app_settings:
|
||||
to_be_updated = True
|
||||
self.app_settings_strDic.properties = dict()
|
||||
|
||||
# check if app settings changed
|
||||
if self.purge_app_settings or self.is_app_settings_changed():
|
||||
to_be_updated = True
|
||||
self.to_do = Actions.CreateOrUpdate
|
||||
|
||||
if self.app_settings:
|
||||
for key in self.app_settings.keys():
|
||||
self.app_settings_strDic.properties[key] = self.app_settings[key]
|
||||
|
||||
elif self.state == 'absent':
|
||||
if old_response:
|
||||
self.log("Delete Web App instance")
|
||||
self.results['changed'] = True
|
||||
|
||||
if self.check_mode:
|
||||
return self.results
|
||||
|
||||
self.delete_webapp()
|
||||
|
||||
self.log('Web App instance deleted')
|
||||
|
||||
else:
|
||||
self.fail("Web app {0} not exists.".format(self.name))
|
||||
|
||||
if to_be_updated:
|
||||
self.log('Need to Create/Update web app')
|
||||
self.results['changed'] = True
|
||||
|
||||
if self.check_mode:
|
||||
return self.results
|
||||
|
||||
if self.to_do == Actions.CreateOrUpdate:
|
||||
response = self.create_update_webapp()
|
||||
self.results['id'] = response['id']
|
||||
|
||||
return self.results
|
||||
|
||||
# compare existing web app with input, determine weather it's update operation
|
||||
def is_updatable_property_changed(self, existing_webapp):
|
||||
for property_name in self.updatable_properties:
|
||||
if hasattr(self, property_name) and getattr(self, property_name) != existing_webapp.get(property_name, None):
|
||||
return True
|
||||
|
||||
return False
|
||||
|
||||
# compare xxx_version
|
||||
def is_site_config_changed(self, existing_config):
|
||||
for fx_version in self.site_config_updatable_properties:
|
||||
if self.site_config.get(fx_version):
|
||||
if not getattr(existing_config, fx_version) or \
|
||||
getattr(existing_config, fx_version).upper() != self.site_config.get(fx_version).upper():
|
||||
return True
|
||||
|
||||
return False
|
||||
|
||||
# comparing existing app setting with input, determine whether it's changed
|
||||
def is_app_settings_changed(self):
|
||||
if self.app_settings:
|
||||
if len(self.app_settings_strDic.properties) != len(self.app_settings):
|
||||
return True
|
||||
|
||||
elif self.app_settings_strDic.properties and len(self.app_settings_strDic.properties) > 0:
|
||||
for key in self.app_settings.keys():
|
||||
if not self.app_settings_strDic.properties.get(key) \
|
||||
or self.app_settings[key] != self.app_settings_strDic.properties[key]:
|
||||
return True
|
||||
return False
|
||||
|
||||
# comparing deployment source with input, determine wheather it's changed
|
||||
def is_deployment_source_changed(self, existing_webapp):
|
||||
if self.deployment_source:
|
||||
if self.deployment_source.get('url') \
|
||||
and self.deployment_source['url'] != existing_webapp.get('site_source_control')['url']:
|
||||
return True
|
||||
|
||||
if self.deployment_source.get('branch') \
|
||||
and self.deployment_source['branch'] != existing_webapp.get('site_source_control')['branch']:
|
||||
return True
|
||||
|
||||
return False
|
||||
|
||||
def create_update_webapp(self):
|
||||
'''
|
||||
Creates or updates Web App with the specified configuration.
|
||||
|
||||
:return: deserialized Web App instance state dictionary
|
||||
'''
|
||||
self.log(
|
||||
"Creating / Updating the Web App instance {0}".format(self.name))
|
||||
|
||||
try:
|
||||
skip_dns_registration = self.dns_registration
|
||||
force_dns_registration = None if self.dns_registration is None else not self.dns_registration
|
||||
|
||||
response = self.web_client.web_apps.create_or_update(resource_group_name=self.resource_group,
|
||||
name=self.name,
|
||||
site_envelope=self.site,
|
||||
skip_dns_registration=skip_dns_registration,
|
||||
skip_custom_domain_verification=self.skip_custom_domain_verification,
|
||||
force_dns_registration=force_dns_registration,
|
||||
ttl_in_seconds=self.ttl_in_seconds)
|
||||
if isinstance(response, AzureOperationPoller):
|
||||
response = self.get_poller_result(response)
|
||||
|
||||
except CloudError as exc:
|
||||
self.log('Error attempting to create the Web App instance.')
|
||||
self.fail(
|
||||
"Error creating the Web App instance: {0}".format(str(exc)))
|
||||
return response.as_dict()
|
||||
|
||||
def delete_webapp(self):
|
||||
'''
|
||||
Deletes specified Web App instance in the specified subscription and resource group.
|
||||
|
||||
:return: True
|
||||
'''
|
||||
self.log("Deleting the Web App instance {0}".format(self.name))
|
||||
try:
|
||||
response = self.web_client.web_apps.delete(resource_group_name=self.resource_group,
|
||||
name=self.name)
|
||||
except CloudError as e:
|
||||
self.log('Error attempting to delete the Web App instance.')
|
||||
self.fail(
|
||||
"Error deleting the Web App instance: {0}".format(str(e)))
|
||||
|
||||
return True
|
||||
|
||||
def get_webapp(self):
|
||||
'''
|
||||
Gets the properties of the specified Web App.
|
||||
|
||||
:return: deserialized Web App instance state dictionary
|
||||
'''
|
||||
self.log(
|
||||
"Checking if the Web App instance {0} is present".format(self.name))
|
||||
|
||||
response = None
|
||||
|
||||
try:
|
||||
response = self.web_client.web_apps.get(resource_group_name=self.resource_group,
|
||||
name=self.name)
|
||||
|
||||
self.log("Response : {0}".format(response))
|
||||
self.log("Web App instance : {0} found".format(response.name))
|
||||
return response.as_dict()
|
||||
|
||||
except CloudError as ex:
|
||||
self.log("Didn't find web app {0} in resource group {1}".format(
|
||||
self.name, self.resource_group))
|
||||
|
||||
return False
|
||||
|
||||
def get_app_service_plan(self):
|
||||
'''
|
||||
Gets app service plan
|
||||
:return: deserialized app service plan dictionary
|
||||
'''
|
||||
self.log("Get App Service Plan {0}".format(self.plan['name']))
|
||||
|
||||
try:
|
||||
response = self.web_client.app_service_plans.get(
|
||||
self.plan['resource_group'], self.plan['name'])
|
||||
self.log("Response : {0}".format(response))
|
||||
self.log("App Service Plan : {0} found".format(response.name))
|
||||
|
||||
return response.as_dict()
|
||||
except CloudError as ex:
|
||||
self.log("Didn't find app service plan {0} in resource group {1}".format(
|
||||
self.plan['name'], self.plan['resource_group']))
|
||||
|
||||
return False
|
||||
|
||||
def create_app_service_plan(self):
|
||||
'''
|
||||
Creates app service plan
|
||||
:return: deserialized app service plan dictionary
|
||||
'''
|
||||
self.log("Create App Service Plan {0}".format(self.plan['name']))
|
||||
|
||||
try:
|
||||
# normalize sku
|
||||
sku = _normalize_sku(self.plan['sku'])
|
||||
|
||||
sku_def = SkuDescription(tier=get_sku_name(
|
||||
sku), name=sku, capacity=(self.plan.get('number_of_workers', None)))
|
||||
plan_def = AppServicePlan(
|
||||
location=self.plan['location'], app_service_plan_name=self.plan['name'], sku=sku_def, reserved=(self.plan.get('is_linux', None)))
|
||||
|
||||
poller = self.web_client.app_service_plans.create_or_update(
|
||||
self.plan['resource_group'], self.plan['name'], plan_def)
|
||||
|
||||
if isinstance(poller, AzureOperationPoller):
|
||||
response = self.get_poller_result(poller)
|
||||
|
||||
self.log("Response : {0}".format(response))
|
||||
|
||||
return response.as_dict()
|
||||
except CloudError as ex:
|
||||
self.fail("Failed to create app service plan {0} in resource group {1}: {2}".format(
|
||||
self.plan['name'], self.plan['resource_group'], str(ex)))
|
||||
|
||||
def list_app_settings(self):
|
||||
'''
|
||||
List application settings
|
||||
:return: deserialized list response
|
||||
'''
|
||||
self.log("List application setting")
|
||||
|
||||
try:
|
||||
|
||||
response = self.web_client.web_apps.list_application_settings(
|
||||
resource_group_name=self.resource_group, name=self.name)
|
||||
self.log("Response : {0}".format(response))
|
||||
|
||||
return response
|
||||
except CloudError as ex:
|
||||
self.log("Failed to list application settings for web app {0} in resource group {1}".format(
|
||||
self.name, self.resource_group))
|
||||
|
||||
return False
|
||||
|
||||
def update_app_settings(self):
|
||||
'''
|
||||
Update application settings
|
||||
:return: deserialized updating response
|
||||
'''
|
||||
self.log("Update application setting")
|
||||
|
||||
try:
|
||||
response = self.web_client.web_apps.update_application_settings(
|
||||
resource_group_name=self.resource_group, name=self.name, app_settings=self.app_settings_strDic)
|
||||
self.log("Response : {0}".format(response))
|
||||
|
||||
return response.as_dict()
|
||||
except CloudError as ex:
|
||||
self.log("Failed to update application settings for web app {0} in resource group {1}".format(
|
||||
self.name, self.resource_group))
|
||||
|
||||
return False
|
||||
|
||||
def create_or_update_source_control(self):
|
||||
'''
|
||||
Update site source control
|
||||
:return: deserialized updating response
|
||||
'''
|
||||
self.log("Update site source control")
|
||||
|
||||
if self.deployment_source is None:
|
||||
return False
|
||||
|
||||
self.deployment_source['is_manual_integration'] = False
|
||||
self.deployment_source['is_mercurial'] = False
|
||||
|
||||
try:
|
||||
response = self.web_client.web_client.create_or_update_source_control(
|
||||
self.resource_group, self.name, self.deployment_source)
|
||||
self.log("Response : {0}".format(response))
|
||||
|
||||
return response.as_dict()
|
||||
except CloudError as ex:
|
||||
self.fail("Failed to update site source control for web app {0} in resource group {1}".format(
|
||||
self.name, self.resource_group))
|
||||
|
||||
def get_webapp_configuration(self):
|
||||
'''
|
||||
Get web app configuration
|
||||
:return: deserialized web app configuration response
|
||||
'''
|
||||
self.log("Get web app configuration")
|
||||
|
||||
try:
|
||||
|
||||
response = self.web_client.web_apps.get_configuration(
|
||||
resource_group_name=self.resource_group, name=self.name)
|
||||
self.log("Response : {0}".format(response))
|
||||
|
||||
return response
|
||||
except CloudError as ex:
|
||||
self.log("Failed to get configuration for web app {0} in resource group {1}: {2}".format(
|
||||
self.name, self.resource_group, str(ex)))
|
||||
|
||||
return False
|
||||
|
||||
|
||||
def main():
|
||||
"""Main execution"""
|
||||
AzureRMWebApps()
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
3
test/integration/targets/azure_rm_webapp/aliases
Normal file
3
test/integration/targets/azure_rm_webapp/aliases
Normal file
|
@ -0,0 +1,3 @@
|
|||
cloud/azure
|
||||
shippable/azure/group4
|
||||
destructive
|
2
test/integration/targets/azure_rm_webapp/meta/main.yml
Normal file
2
test/integration/targets/azure_rm_webapp/meta/main.yml
Normal file
|
@ -0,0 +1,2 @@
|
|||
dependencies:
|
||||
- setup_azure
|
235
test/integration/targets/azure_rm_webapp/tasks/main.yml
Normal file
235
test/integration/targets/azure_rm_webapp/tasks/main.yml
Normal file
|
@ -0,0 +1,235 @@
|
|||
- name: Fix resource prefix
|
||||
set_fact:
|
||||
linux_app_plan_resource_group: "{{ resource_group_secondary }}"
|
||||
win_app_name: "{{ (resource_prefix | replace('-','x'))[-8:] }}{{ 1000 | random}}winapp"
|
||||
win_plan_name: "{{ (resource_prefix | replace('-','x'))[-8:] }}winplan"
|
||||
linux_plan_name: "{{ (resource_group_secondary | replace('-','x'))[-8:] }}linplan"
|
||||
|
||||
- name: Create a windows web app with non-exist app service plan
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}1"
|
||||
plan:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_plan_name }}"
|
||||
is_linux: false
|
||||
sku: S1
|
||||
|
||||
- name: Create a windows web app with existing app service plan
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}2"
|
||||
plan: "{{ win_plan_name }}"
|
||||
register: output
|
||||
|
||||
- name: Create a windows web app with existing app service plan, try to update some root level params
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}3"
|
||||
plan: "{{ win_plan_name }}"
|
||||
dns_registration: true
|
||||
https_only: true
|
||||
register: output
|
||||
|
||||
- name: Create a win web app with java run time specific
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}4"
|
||||
plan: "{{ win_plan_name }}"
|
||||
frameworks:
|
||||
- name: "java"
|
||||
version: "1.8"
|
||||
settings:
|
||||
java_container: "Tomcat"
|
||||
java_container_version: "8.0"
|
||||
app_settings:
|
||||
testkey: "testvalue"
|
||||
register: output
|
||||
|
||||
- name: assert the function was created
|
||||
assert:
|
||||
that: output.changed
|
||||
|
||||
- name: Update app settings
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}4"
|
||||
plan: "{{ win_plan_name }}"
|
||||
frameworks:
|
||||
- name: "java"
|
||||
version: "1.8"
|
||||
settings:
|
||||
java_container: "Tomcat"
|
||||
java_container_version: "8.0"
|
||||
app_settings:
|
||||
testkey2: "testvalue2"
|
||||
register: output
|
||||
|
||||
- name: Assert the web app was created
|
||||
assert:
|
||||
that: output.changed
|
||||
|
||||
- name: Purge all existing app settings
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}4"
|
||||
plan: "{{ win_plan_name }}"
|
||||
purge_app_settings: true
|
||||
register: output
|
||||
|
||||
- name: Assert the web app was created
|
||||
assert:
|
||||
that: output.changed
|
||||
|
||||
- name: Create a win web app with python run time and php run time
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}5"
|
||||
plan: "{{ win_plan_name }}"
|
||||
frameworks:
|
||||
- name: "python"
|
||||
version: "2.7"
|
||||
- name: node
|
||||
version: "6.6"
|
||||
- name: "php"
|
||||
version: "7.0"
|
||||
register: output
|
||||
|
||||
- name: Assert the web app was created
|
||||
assert:
|
||||
that: output.changed
|
||||
|
||||
- name: Create a docker web app with some app settings
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}6"
|
||||
plan:
|
||||
resource_group: "{{ linux_app_plan_resource_group }}"
|
||||
name: "{{ linux_plan_name }}"
|
||||
is_linux: true
|
||||
sku: S1
|
||||
number_of_workers: 1
|
||||
container_settings:
|
||||
name: "ansible/ansible:ubuntu1404"
|
||||
register: output
|
||||
|
||||
- name: Assert the web app was created
|
||||
assert:
|
||||
that: output.changed
|
||||
|
||||
- name: Create a docker web app with private acr registry
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}7"
|
||||
plan:
|
||||
resource_group: "{{ linux_app_plan_resource_group }}"
|
||||
name: "{{ linux_plan_name }}"
|
||||
container_settings:
|
||||
name: "ansible/ansible:ubuntu1404"
|
||||
registry_server_url: test.io
|
||||
registry_server_user: user
|
||||
registry_server_password: password
|
||||
register: output
|
||||
|
||||
- name: Assert the web app was created
|
||||
assert:
|
||||
that: output.changed
|
||||
|
||||
- name: Create a linux web app with nodejs framework
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}8"
|
||||
plan:
|
||||
resource_group: "{{ linux_app_plan_resource_group }}"
|
||||
name: "{{ linux_plan_name }}"
|
||||
frameworks:
|
||||
- name: node
|
||||
version: "6.6"
|
||||
register: output
|
||||
|
||||
- name: Should be idempotent with linux web app created
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}8"
|
||||
plan:
|
||||
resource_group: "{{ linux_app_plan_resource_group }}"
|
||||
name: "{{ linux_plan_name }}"
|
||||
frameworks:
|
||||
- name: node
|
||||
version: "6.6"
|
||||
register: output
|
||||
|
||||
- assert:
|
||||
that: not output.changed
|
||||
|
||||
- name: Update nodejs framework
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}8"
|
||||
plan:
|
||||
resource_group: "{{ linux_app_plan_resource_group }}"
|
||||
name: "{{ linux_plan_name }}"
|
||||
frameworks:
|
||||
- name: node
|
||||
version: "6.9"
|
||||
register: output
|
||||
|
||||
- name: Assert the web app was created
|
||||
assert:
|
||||
that: output.changed
|
||||
|
||||
- name: Create a linux web app with deployment source github
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}10"
|
||||
plan:
|
||||
resource_group: "{{ linux_app_plan_resource_group }}"
|
||||
name: "{{ linux_plan_name }}"
|
||||
deployment_source:
|
||||
url: "https://github.com/test/test"
|
||||
branch: master
|
||||
scm_type: GitHub
|
||||
register: output
|
||||
|
||||
- name: Assert the web app was created
|
||||
assert:
|
||||
that: output.changed
|
||||
|
||||
- name: Delete web app
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}8"
|
||||
state: absent
|
||||
register: output
|
||||
|
||||
- name: Assert the web app was deleted
|
||||
assert:
|
||||
that: output.changed
|
||||
|
||||
- name: assert error that java is mutually exclusive with frameworks
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ win_app_name }}11"
|
||||
plan: "{{ win_plan_name }}"
|
||||
frameworks:
|
||||
- name: "python"
|
||||
version: "2.7"
|
||||
- name: "java"
|
||||
version: "1.8"
|
||||
register: fail_win_java_version_mutual_exclusive
|
||||
failed_when: 'fail_win_java_version_mutual_exclusive.msg != "Java is mutually exclusive with other frameworks."'
|
||||
|
||||
- name: assert error when linux web app, only can specify one framework
|
||||
azure_rm_webapp:
|
||||
resource_group: "{{ resource_group }}"
|
||||
name: "{{ linux_plan_name }}12"
|
||||
plan:
|
||||
resource_group: "{{ linux_app_plan_resource_group }}"
|
||||
name: "{{ linux_plan_name }}"
|
||||
frameworks:
|
||||
- name: "python"
|
||||
version: "2.7"
|
||||
- name: "node"
|
||||
version: "6.6"
|
||||
register: fail_linux_one_framework_only
|
||||
failed_when: fail_linux_one_framework_only.msg != "Can specify one framework only for Linux web app."
|
Loading…
Reference in a new issue