Bladeren bron

Merge pull request #8584 from tomassedovic/openstack-allow-empty-clusterid

Fix OpenStack nsupdate zone handling
OpenShift Merge Robot 6 jaren geleden
bovenliggende
commit
df4784d0d7

+ 138 - 28
playbooks/openstack/configuration.md

@@ -120,53 +120,163 @@ your deployment, you must set `OPENSHIFT_CLUSTER` to your stack name to avoid er
 
 
 ## DNS Configuration
 ## DNS Configuration
 
 
-Note that the provisioning playbooks update the original Neutron subnet
-created with the Heat stack to point to the configured DNS servers.
-So the provisioned cluster nodes will start using those natively as
-default nameservers. Technically, this allows the deployment of OpenShift
-clusters without dnsmasq proxies.
+OpenStack deployments require an external DNS server for now. This
+server must be able to resolve the the OpenShift node names to their
+internal IP addresses.
 
 
-In `inventory/group_vars/all.yml`:
+We will be looking into using the internal Neutron DNS and/or the
+Designate project in the future.
+
+While we do not create a DNS for you, if it supports nsupdate (RFC
+2136[nsupdate-rfc]), we can populate it with the cluster records
+automatically.
+
+[nsupdate-rfc]: https://www.ietf.org/rfc/rfc2136.txt
+
+### OpenShift Cluster Domain
+
+To set up the domain name of your OpenShift cluster, set these
+parameters in `inventory/group_vars/all.yml`:
 
 
 * `openshift_openstack_clusterid` Defaults to `openshift`
 * `openshift_openstack_clusterid` Defaults to `openshift`
 * `openshift_openstack_public_dns_domain` Defaults to `example.com`
 * `openshift_openstack_public_dns_domain` Defaults to `example.com`
 
 
-These two parameters together form the cluster's public DNS domain that all
-the servers will be under; by default this domain will be `openshift.example.com`.
+Together, they form the cluster's public DNS domain that all the
+servers will be under; by default this domain will be
+`openshift.example.com`.
 
 
-* `openshift_openstack_app_subdomain` Subdomain for workloads. Defaults to `apps`.
+They're split so you can deploy multiple clusters under the same
+domain with a single inventory change: e.g. `testing.example.com` and
+`production.example.com`.
 
 
-* `openshift_openstack_public_hostname_suffix` Empty by default.
-* `openshift_openstack_private_hostname_suffix` Empty by default.
+You will also want to put the IP addresses of your DNS server(s) in
+the `openshift_openstack_dns_nameservers` array in the same file.
+
+This will configure the Neutron subnet with all the OpenShift nodes to forward
+to these DNS servers. Which means that any server running in that subnet will
+use the DNS automatically, without any extra configuration.
 
 
-If you want to use two sets of hostnames for public and private/prefixed DNS
-records for your externally managed public DNS server, you can specify the
-`openshift_openstack_*_hostname_suffix` parameters. These suffixes are added to
-the nsupdate records sent to the external DNS server. Note that the real hostnames,
-Nova servers, and ansible hostnames and inventory variables are not be updated.
-The deployment may be done on arbitrary named hosts with the hostnames managed by
-cloud-init. Inventory hostnames will ignore these suffixes.
 
 
-* `openshift_openstack_dns_nameservers` List of DNS servers accessible from all the created Nova servers. These will provide the internal name resolution for your OpenShift nodes (as well as upstream name resolution for installing packages, etc.).
+### Adding the DNS Records Automatically
 
 
-* `openshift_use_dnsmasq` Controls whether dnsmasq is deployed or not.By default, dnsmasq is deployed and comes as the hosts' /etc/resolv.conf file first nameserver entry that points to the local host instance of the dnsmasq daemon that in turn proxies DNS requests to the authoritative DNS server. When Network Manager is enabled for provisioned cluster nodes, which is normally the case, you should not change the defaults and always deploy dnsmasq.
+If your DNS supports nsupdate, you can set up the
+`openshift_openstack_external_nsupdate_keys` variable and all the
+necessary DNS records will be added during the provisioning phase
+(after the OpenShift nodes are created, but before we install anything
+on them).
 
 
-* `openshift_openstack_external_nsupdate_keys` Describes an external authoritative DNS server(s) processing dynamic records updates in the public only cluster view. For example:
+Add this to your `inventory/group_vars/all.yml`:
 
 
 ```
 ```
     openshift_openstack_external_nsupdate_keys:
     openshift_openstack_external_nsupdate_keys:
-      public:
+      private:
         key_secret: <some nsupdate key>
         key_secret: <some nsupdate key>
         key_algorithm: 'hmac-md5'
         key_algorithm: 'hmac-md5'
         key_name: 'update-key'
         key_name: 'update-key'
-        server: <public DNS server IP>
+        server: <private DNS server IP>
+```
+
+Make sure that all four values (key secret, algorithm, key name and
+the DNS IP address) are correct.
+
+This will create the records for the internal OpenShift communication.
+If you also want public records for external access, add another
+section called `public` with the same structure.
+
+If you want to use the same DNS server for both public and private
+records, you must set at least one of:
+
+* `openshift_openstack_public_hostname_suffix` Empty by default.
+* `openshift_openstack_private_hostname_suffix` Empty by default.
+
+Otherwise the private records will be overwritten by the public ones.
+
+For example by leaving the *private* suffix empty and setting the *public* one
+to:
+
+```
+openshift_openstack_public_hostname_suffix: -public
+```
+
+The internal access to the first master node would be available with:
+`master-0.openshift.example.com`, while the public access using the floating IP
+address would be under `master-0-public.openshift.example.com`.
+
+Note that these suffixes are only applied to the OpenShift Node names
+as they appear in the DNS. They will not affect the actual hostnames.
+
+It is recommended that you use two separate servers for the private
+and public access instead.
+
+If your nsupdate zone differs from the full OpenShift DNS name (e.g.
+your DNS' zone is "example.com" but you want your cluster to be at
+"openshift.example.com"), you can specify the zone in this parameter:
+
+* `openshift_openstack_nsupdate_zone: example.com`
+
+If left out, it will be equal to the OpenShift cluster DNS.
+
+Don't forget to put your the internal (private) DNS servers to the
+`openshift_openstack_dns_nameservers` array.
+
+
+### Custom DNS Records Configuration
+
+If you're unable (or do not want) to use nsupdate, you will have to
+create your DNS records out-of-band.
+
+To do that, you will have to split the deployment into three phases:
+
+1. Provision (creates the OpenShift servers)
+2. Create DNS records (this is your responsibility)
+3. Installation (installs OpenShift on the servers)
+
+To do this, run the `provision.yml` and `install.yml` playbooks
+instead of the all-in-one `provision_install.yml` and add your DNS
+records between the runs.
+
+You still need to set the `openshift_openstack_dns_nameservers` with
+your (private/internal) DNS servers in `inventory/group_vars/all.yml`.
+
+Next, you need to create a DNS record for every OpenShift node that
+was created. This record must point to the node's **private** IP
+address (not the floating IP).
+
+You can see the server names and their private floating IP addresses
+by running `openstack server list`.
+
+For example with the following output:
+
+```
+$ openstack server list
++--------------------------------------+--------------------------------------+---------+----------------------------------------------------------------------------+---------+-----------+
+| ID                                   | Name                                 | Status  | Networks                                                                   | Image   | Flavor    |
++--------------------------------------+--------------------------------------+---------+----------------------------------------------------------------------------+---------+-----------+
+| 8445bd74-aaf1-4c54-b6fe-e98efa6e47de | master-0.openshift.example.com     | ACTIVE  | openshift-ansible-openshift.example.com-net=192.168.99.10, 10.40.128.136 | centos7 | m1.medium |
+| 635f0a24-bde7-488d-aa0d-c31e0a01e7c4 | infra-node-0.openshift.example.com | ACTIVE  | openshift-ansible-openshift.example.com-net=192.168.99.4, 10.40.128.130  | centos7 | m1.medium |
+| 04657a99-29b1-48c8-8979-3c88ee1c1615 | app-node-0.openshift.example.com   | ACTIVE  | openshift-ansible-openshift.example.com-net=192.168.99.6, 10.40.128.132  | centos7 | m1.medium |
++--------------------------------------+--------------------------------------+---------+----------------------------------------------------------------------------+---------+-----------+
+```
+
+You will need to create these A records:
+
+```
+master-0.openshift.cool.       192.168.99.10
+infra-node-0.openshift.cool.   192.168.99.4
+app-node-0.openshift.cool.     192.168.99.16
+```
+
+For the public access, you'll need to create 2 records: one for the
+API access and the other for the OpenShift apps running on the
+cluster.
+
+```
+console.openshift.cool.    10.40.128.137
+*.apps.openshift.cool.     10.40.128.129
 ```
 ```
 
 
-Here, for the public view section, we specified another key algorithm and
-optional `key_name`, which normally defaults to the cluster's DNS domain.
-This just illustrates a compatibility mode with a DNS service deployed
-by OpenShift on OSP10 reference architecture, and used in a mixed mode with
-another external DNS server.
+These must point to the publicly-accessible IP addresses of your
+master and infra nodes or preferably to the load balancers.
 
 
 
 
 ## Kuryr Networking Configuration
 ## Kuryr Networking Configuration

+ 15 - 2
playbooks/openstack/sample-inventory/group_vars/OSEv3.yml

@@ -4,9 +4,22 @@ openshift_deployment_type: origin
 #openshift_repos_enable_testing: true
 #openshift_repos_enable_testing: true
 #openshift_deployment_type: openshift-enterprise
 #openshift_deployment_type: openshift-enterprise
 #openshift_release: v3.5
 #openshift_release: v3.5
-openshift_master_default_subdomain: "apps.{{ openshift_openstack_clusterid }}.{{ openshift_openstack_public_dns_domain }}"
 
 
-openshift_master_cluster_public_hostname: "console.{{ openshift_openstack_clusterid }}.{{ openshift_openstack_public_dns_domain }}"
+# Default domain to access the applications running on OpenShift.
+# This uses the `openshift_openstack_clusterid`
+# and `openshift_openstack_public_dns_domain` values from all.yml.
+# It will be set to `apps.openshift.example.com` by default.
+# Feel free to change this to a value you prefer. It should be under the
+# domain the OpenShift cluster is configured, though.
+openshift_master_default_subdomain: "apps.{{ (openshift_openstack_clusterid|trim == '') | ternary(openshift_openstack_public_dns_domain, openshift_openstack_clusterid + '.' + openshift_openstack_public_dns_domain) }}"
+
+# Domain to access the OpenShift UI and API.
+# This uses the `openshift_openstack_clusterid`
+# and `openshift_openstack_public_dns_domain` values from all.yml.
+# It will be set to `console.openshift.example.com` by default.
+# Feel free to change this to a value you prefer. It should be under the
+# domain the OpenShift cluster is configured, though.
+openshift_master_cluster_public_hostname: "console.{{ (openshift_openstack_clusterid|trim == '') | ternary(openshift_openstack_public_dns_domain, openshift_openstack_clusterid + '.' + openshift_openstack_public_dns_domain) }}"
 
 
 osm_default_node_selector: 'region=primary'
 osm_default_node_selector: 'region=primary'
 
 

+ 1 - 1
roles/openshift_openstack/defaults/main.yml

@@ -51,7 +51,7 @@ openshift_openstack_public_hostname_suffix: ""
 openshift_openstack_private_hostname_suffix: ""
 openshift_openstack_private_hostname_suffix: ""
 openshift_openstack_public_dns_domain: "example.com"
 openshift_openstack_public_dns_domain: "example.com"
 openshift_openstack_full_dns_domain: "{{ (openshift_openstack_clusterid|trim == '') | ternary(openshift_openstack_public_dns_domain, openshift_openstack_clusterid + '.' + openshift_openstack_public_dns_domain) }}"
 openshift_openstack_full_dns_domain: "{{ (openshift_openstack_clusterid|trim == '') | ternary(openshift_openstack_public_dns_domain, openshift_openstack_clusterid + '.' + openshift_openstack_public_dns_domain) }}"
-openshift_openstack_app_subdomain: "apps"
+openshift_openstack_nsupdate_zone: "{{ openshift_openstack_full_dns_domain }}"
 
 
 
 
 # heat vars
 # heat vars

+ 10 - 0
roles/openshift_openstack/tasks/check-prerequisites.yml

@@ -117,3 +117,13 @@
       by your cloud.
       by your cloud.
   when: (openshift_openstack_use_lbaas_load_balancer and openshift_openstack_lbaasv2_provider == 'Neutron::LBaaS' and 'lbaasv2' not in openstack_network_extensions) or
   when: (openshift_openstack_use_lbaas_load_balancer and openshift_openstack_lbaasv2_provider == 'Neutron::LBaaS' and 'lbaasv2' not in openstack_network_extensions) or
         (openshift_openstack_use_lbaas_load_balancer and openshift_openstack_lbaasv2_provider == 'Octavia' and 'load-balancer' not in openstack_service_catalog)
         (openshift_openstack_use_lbaas_load_balancer and openshift_openstack_lbaasv2_provider == 'Octavia' and 'load-balancer' not in openstack_service_catalog)
+
+- name: Verify the nsupdate zone is a subset of the full cluster domain
+  fail:
+    msg: >
+      The `openshift_openstack_nsupdate_zone` ({{ openshift_openstack_nsupdate_zone }})
+      must be a substring of `openshift_openstack_full_dns_domain ({{ openshift_openstack_full_dns_domain }})`
+  when:
+  - openshift_openstack_nsupdate_zone is defined
+  - openshift_openstack_full_dns_domain is defined
+  - openshift_openstack_full_dns_domain is not match(".*" + openshift_openstack_nsupdate_zone + "$")

+ 1 - 1
roles/openshift_openstack/tasks/clean-dns.yml

@@ -9,7 +9,7 @@
     key_algorithm: "{{ item.0.key_algorithm }}"
     key_algorithm: "{{ item.0.key_algorithm }}"
     server: "{{ item.0.server }}"
     server: "{{ item.0.server }}"
     zone: "{{ item.0.zone }}"
     zone: "{{ item.0.zone }}"
-    record: "{{ item.1.hostname }}"
+    record: "{{ item.1.fqdn | replace('.' + openshift_openstack_nsupdate_zone, '') }}"
     value: "{{ item.1.ip }}"
     value: "{{ item.1.ip }}"
     type: "{{ item.1.type }}"
     type: "{{ item.1.type }}"
     state: absent
     state: absent

+ 15 - 13
roles/openshift_openstack/tasks/generate-dns.yml

@@ -1,24 +1,25 @@
 ---
 ---
 - name: "Generate list of private A records"
 - name: "Generate list of private A records"
   set_fact:
   set_fact:
-    private_records: "{{ private_records | default([]) + [ { 'type': 'A', 'hostname': hostvars[item]['ansible_hostname'] + openshift_openstack_private_hostname_suffix, 'ip': hostvars[item]['private_v4'] } ] }}"
+    private_records: "{{ private_records | default([]) + [ { 'type': 'A', 'fqdn': hostvars[item]['ansible_hostname'] + openshift_openstack_private_hostname_suffix + '.' + openshift_openstack_full_dns_domain, 'ip': hostvars[item]['private_v4'] } ] }}"
   with_items: "{{ groups['cluster_hosts'] }}"
   with_items: "{{ groups['cluster_hosts'] }}"
 
 
 - name: "Add wildcard records to the private A records for infrahosts"
 - name: "Add wildcard records to the private A records for infrahosts"
   set_fact:
   set_fact:
-    private_records: "{{ private_records | default([]) + [ { 'type': 'A', 'hostname': '*.' + openshift_openstack_app_subdomain, 'ip': hostvars[item]['private_v4'] } ] }}"
+    private_records: "{{ private_records | default([]) + [ { 'type': 'A', 'fqdn': '*.' + hostvars[groups.masters[0]].openshift_master_default_subdomain, 'ip': hostvars[item]['private_v4'] } ] }}"
   with_items: "{{ groups['infra_hosts'] }}"
   with_items: "{{ groups['infra_hosts'] }}"
+  when: openshift_openstack_public_router_ip is defined
 
 
 - name: "Add public master cluster hostname records to the private A records (single master)"
 - name: "Add public master cluster hostname records to the private A records (single master)"
   set_fact:
   set_fact:
-    private_records: "{{ private_records | default([]) + [ { 'type': 'A', 'hostname': (hostvars[groups.masters[0]].openshift_master_cluster_public_hostname | replace(openshift_openstack_full_dns_domain, ''))[:-1], 'ip': hostvars[groups.masters[0]].private_v4 } ] }}"
+    private_records: "{{ private_records | default([]) + [ { 'type': 'A', 'fqdn': hostvars[groups.masters[0]].openshift_master_cluster_public_hostname, 'ip': hostvars[groups.masters[0]].private_v4 } ] }}"
   when:
   when:
     - hostvars[groups.masters[0]].openshift_master_cluster_public_hostname is defined
     - hostvars[groups.masters[0]].openshift_master_cluster_public_hostname is defined
     - openshift_openstack_num_masters == 1
     - openshift_openstack_num_masters == 1
 
 
 - name: "Add public master cluster hostname records to the private A records (multi-master)"
 - name: "Add public master cluster hostname records to the private A records (multi-master)"
   set_fact:
   set_fact:
-    private_records: "{{ private_records | default([]) + [ { 'type': 'A', 'hostname': (hostvars[groups.masters[0]].openshift_master_cluster_public_hostname | replace(openshift_openstack_full_dns_domain, ''))[:-1], 'ip': hostvars[groups.lb[0]].private_v4 } ] }}"
+    private_records: "{{ private_records | default([]) + [ { 'type': 'A', 'fqdn': hostvars[groups.masters[0]].openshift_master_cluster_public_hostname, 'ip': hostvars[groups.lb[0]].private_v4 } ] }}"
   when:
   when:
     - hostvars[groups.masters[0]].openshift_master_cluster_public_hostname is defined
     - hostvars[groups.masters[0]].openshift_master_cluster_public_hostname is defined
     - openshift_openstack_num_masters > 1
     - openshift_openstack_num_masters > 1
@@ -28,7 +29,7 @@
     nsupdate_server_private: "{{ openshift_openstack_external_nsupdate_keys['private']['server'] }}"
     nsupdate_server_private: "{{ openshift_openstack_external_nsupdate_keys['private']['server'] }}"
     nsupdate_key_secret_private: "{{ openshift_openstack_external_nsupdate_keys['private']['key_secret'] }}"
     nsupdate_key_secret_private: "{{ openshift_openstack_external_nsupdate_keys['private']['key_secret'] }}"
     nsupdate_key_algorithm_private: "{{ openshift_openstack_external_nsupdate_keys['private']['key_algorithm'] }}"
     nsupdate_key_algorithm_private: "{{ openshift_openstack_external_nsupdate_keys['private']['key_algorithm'] }}"
-    nsupdate_private_key_name: "{{ openshift_openstack_external_nsupdate_keys['private']['key_name']|default('private-' + openshift_openstack_full_dns_domain) }}"
+    nsupdate_private_key_name: "{{ openshift_openstack_external_nsupdate_keys['private']['key_name'] }}"
   when:
   when:
     - openshift_openstack_external_nsupdate_keys['private'] is defined
     - openshift_openstack_external_nsupdate_keys['private'] is defined
 
 
@@ -37,9 +38,9 @@
   set_fact:
   set_fact:
     private_named_records:
     private_named_records:
       - view: "private"
       - view: "private"
-        zone: "{{ openshift_openstack_full_dns_domain }}"
+        zone: "{{ openshift_openstack_nsupdate_zone }}"
         server: "{{ nsupdate_server_private }}"
         server: "{{ nsupdate_server_private }}"
-        key_name: "{{ nsupdate_private_key_name|default('private-' + openshift_openstack_full_dns_domain) }}"
+        key_name: "{{ nsupdate_private_key_name }}"
         key_secret: "{{ nsupdate_key_secret_private }}"
         key_secret: "{{ nsupdate_key_secret_private }}"
         key_algorithm: "{{ nsupdate_key_algorithm_private | lower }}"
         key_algorithm: "{{ nsupdate_key_algorithm_private | lower }}"
         entries: "{{ private_records }}"
         entries: "{{ private_records }}"
@@ -48,17 +49,18 @@
 
 
 - name: "Generate list of public A records"
 - name: "Generate list of public A records"
   set_fact:
   set_fact:
-    public_records: "{{ public_records | default([]) + [ { 'type': 'A', 'hostname': hostvars[item]['ansible_hostname'] + openshift_openstack_public_hostname_suffix, 'ip': hostvars[item]['public_v4'] } ] }}"
+    public_records: "{{ public_records | default([]) + [ { 'type': 'A', 'fqdn': hostvars[item]['ansible_hostname'] + openshift_openstack_public_hostname_suffix + '.' + openshift_openstack_full_dns_domain, 'ip': hostvars[item]['public_v4'] } ] }}"
   with_items: "{{ groups['cluster_hosts'] }}"
   with_items: "{{ groups['cluster_hosts'] }}"
   when: hostvars[item]['public_v4'] is defined
   when: hostvars[item]['public_v4'] is defined
 
 
 - name: "Add wildcard record to the public A records"
 - name: "Add wildcard record to the public A records"
   set_fact:
   set_fact:
-    public_records: "{{ public_records | default([]) + [ { 'type': 'A', 'hostname': '*.' + openshift_openstack_app_subdomain, 'ip': openshift_openstack_public_router_ip } ] }}"
+    public_records: "{{ public_records | default([]) + [ { 'type': 'A', 'fqdn': '*.' + hostvars[groups.masters[0]].openshift_master_default_subdomain, 'ip': openshift_openstack_public_router_ip } ] }}"
+  when: openshift_openstack_public_router_ip is defined
 
 
 - name: "Add the public API entry point record"
 - name: "Add the public API entry point record"
   set_fact:
   set_fact:
-    public_records: "{{ public_records | default([]) + [ { 'type': 'A', 'hostname': (hostvars[groups.masters[0]].openshift_master_cluster_public_hostname | replace(openshift_openstack_full_dns_domain, ''))[:-1], 'ip': openshift_openstack_public_api_ip } ] }}"
+    public_records: "{{ public_records | default([]) + [ { 'type': 'A', 'fqdn': hostvars[groups.masters[0]].openshift_master_cluster_public_hostname, 'ip': openshift_openstack_public_api_ip } ] }}"
   when:
   when:
     - hostvars[groups.masters[0]].openshift_master_cluster_public_hostname is defined
     - hostvars[groups.masters[0]].openshift_master_cluster_public_hostname is defined
 
 
@@ -67,7 +69,7 @@
     nsupdate_server_public: "{{ openshift_openstack_external_nsupdate_keys['public']['server'] }}"
     nsupdate_server_public: "{{ openshift_openstack_external_nsupdate_keys['public']['server'] }}"
     nsupdate_key_secret_public: "{{ openshift_openstack_external_nsupdate_keys['public']['key_secret'] }}"
     nsupdate_key_secret_public: "{{ openshift_openstack_external_nsupdate_keys['public']['key_secret'] }}"
     nsupdate_key_algorithm_public: "{{ openshift_openstack_external_nsupdate_keys['public']['key_algorithm'] }}"
     nsupdate_key_algorithm_public: "{{ openshift_openstack_external_nsupdate_keys['public']['key_algorithm'] }}"
-    nsupdate_public_key_name: "{{ openshift_openstack_external_nsupdate_keys['public']['key_name']|default('public-' + openshift_openstack_full_dns_domain) }}"
+    nsupdate_public_key_name: "{{ openshift_openstack_external_nsupdate_keys['public']['key_name'] }}"
   when:
   when:
     - openshift_openstack_external_nsupdate_keys['public'] is defined
     - openshift_openstack_external_nsupdate_keys['public'] is defined
 
 
@@ -75,9 +77,9 @@
   set_fact:
   set_fact:
     public_named_records:
     public_named_records:
       - view: "public"
       - view: "public"
-        zone: "{{ openshift_openstack_full_dns_domain }}"
+        zone: "{{ openshift_openstack_nsupdate_zone }}"
         server: "{{ nsupdate_server_public }}"
         server: "{{ nsupdate_server_public }}"
-        key_name: "{{ nsupdate_public_key_name|default('public-' + openshift_openstack_full_dns_domain) }}"
+        key_name: "{{ nsupdate_public_key_name }}"
         key_secret: "{{ nsupdate_key_secret_public }}"
         key_secret: "{{ nsupdate_key_secret_public }}"
         key_algorithm: "{{ nsupdate_key_algorithm_public | lower }}"
         key_algorithm: "{{ nsupdate_key_algorithm_public | lower }}"
         entries: "{{ public_records }}"
         entries: "{{ public_records }}"

+ 1 - 1
roles/openshift_openstack/tasks/populate-dns.yml

@@ -9,7 +9,7 @@
     key_algorithm: "{{ item.0.key_algorithm }}"
     key_algorithm: "{{ item.0.key_algorithm }}"
     server: "{{ item.0.server }}"
     server: "{{ item.0.server }}"
     zone: "{{ item.0.zone }}"
     zone: "{{ item.0.zone }}"
-    record: "{{ item.1.hostname }}"
+    record: "{{ item.1.fqdn | replace('.' + openshift_openstack_nsupdate_zone, '') }}"
     value: "{{ item.1.ip }}"
     value: "{{ item.1.ip }}"
     type: "{{ item.1.type }}"
     type: "{{ item.1.type }}"
     state: present
     state: present