nuvolaris/storage_static.py (36 lines of code) (raw):
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing,
# software distributed under the License is distributed on an
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
# KIND, either express or implied. See the License for the
# specific language governing permissions and limitations
# under the License.
#
import kopf, logging, json, os
import nuvolaris.kube as kube
import nuvolaris.kustomize as kus
import nuvolaris.config as cfg
import nuvolaris.util as util
import nuvolaris.apihost_util as apihost_util
import nuvolaris.operator_util as operator_util
from nuvolaris.user_config import UserConfig
from nuvolaris.user_metadata import UserMetadata
from nuvolaris.ingress_data import IngressData
from nuvolaris.route_data import RouteData
def create(owner=None):
logging.info("*** configuring nuvolaris nginx static provider")
runtime = cfg.get('nuvolaris.kube')
data = util.get_storage_static_config_data()
data['storage_hostname'] = apihost_util.extract_hostname(data['storage_url'])
logging.info(f"configuring storage static provider to check for availability of endpoint {data['storage_hostname']}")
tplp = ["nginx-static-cm.yaml","nginx-static-sts.yaml","security-set-attach.yaml","set-attach.yaml"]
if(data['affinity'] or data['tolerations']):
tplp.append("affinity-tolerance-sts-core-attach.yaml")
kust = kus.patchTemplates("nginx-static", tplp, data)
spec = kus.kustom_list("nginx-static", kust, templates=[], data=data)
if owner:
kopf.append_owner_reference(spec['items'], owner)
else:
cfg.put("state.nginx-static.spec", spec)
res = kube.apply(spec)
# dynamically detect nginx pod and wait for readiness
util.wait_for_pod_ready("{.items[?(@.metadata.labels.app == 'nuvolaris-static')].metadata.name}")
create_nuv_static_ingress(runtime, owner)
logging.info("*** configured nuvolaris nginx static provider")
return res
def create_nuv_static_ingress(runtime, owner=None):
""""
Deploys the static ingresses for the nuvolaris user
"""
apihost_url = apihost_util.get_apihost(runtime)
hostname = apihost_util.extract_hostname(apihost_url)
should_create_www = "www" not in hostname and runtime not in ["kind"]
bucket_name = util.get_value_from_config_map(path='{.metadata.annotations.s3_bucket_static}')
if runtime == 'openshift':
res = deploy_content_route_template("nuvolaris",bucket_name, apihost_url)
if should_create_www:
res += deploy_content_route_template("www-nuvolaris",bucket_name,apihost_util.append_prefix_to_url(apihost_url,"www"))
return res
else:
res = deploy_content_ingress_template("nuvolaris",bucket_name,apihost_url)
if should_create_www:
res += deploy_content_ingress_template("www-nuvolaris",bucket_name,apihost_util.append_prefix_to_url(apihost_url,"www"))
return res
def static_ingress_name(namespace, default="apihost"):
return namespace == "nuvolaris" and f"{default}-static-ingress" or f"{namespace}-static-ingress"
def static_route_name(namespace):
return f"{namespace}-static-route"
def static_secret_name(namespace):
return f"{namespace}-crt"
def static_middleware_ingress_name(namespace):
return f"{namespace}-static-ingress-add-prefix"
def deploy_content_route_template(namespace, bucket, url):
logging.info(f"**** configuring static openshift route for url {url}")
content = RouteData(url)
content.with_route_name(static_route_name(namespace))
content.with_needs_rewrite(True)
content.with_service_name("nuvolaris-static-svc")
content.with_service_kind("Service")
content.with_service_port("8080")
content.with_context_path("/")
content.with_rewrite_target(f"/{bucket}/")
path_to_template_yaml = content.render_template(namespace)
res = kube.kubectl("apply", "-f",path_to_template_yaml)
os.remove(path_to_template_yaml)
return res
def deploy_content_ingress_template(namespace, bucket, url):
logging.info(f"**** configuring static ingress for url {url}")
content = IngressData(url)
content.with_ingress_name(static_ingress_name(namespace))
content.with_secret_name(static_secret_name(namespace))
content.with_context_path("/")
content.with_context_regexp("(.*)")
content.with_prefix_target(f"/{bucket}")
content.with_service_name("nuvolaris-static-svc")
content.with_service_port("8080")
content.with_middleware_ingress_name(static_middleware_ingress_name(namespace))
res = ""
if content.requires_traefik_middleware():
logging.info("*** configuring traefik middleware")
path_to_template_yaml = content.render_traefik_middleware_template(namespace)
res += kube.kubectl("apply", "-f",path_to_template_yaml)
os.remove(path_to_template_yaml)
logging.info(f"*** configuring static ingress endpoint for {namespace}")
path_to_template_yaml = content.render_template(namespace)
res += kube.kubectl("apply", "-f",path_to_template_yaml)
os.remove(path_to_template_yaml)
return res
def create_ow_static_endpoint(ucfg: UserConfig, user_metadata: UserMetadata, owner=None):
"""
deploy an ingress to access a generic user web bucket
"""
runtime = cfg.get('nuvolaris.kube')
namespace = ucfg.get("namespace")
apihost = ucfg.get("apihost") or "auto"
bucket_name = ucfg.get('S3_BUCKET_STATIC')
hostname = apihost_util.get_user_static_hostname(runtime, namespace, apihost)
logging.debug("using hostname %s to configure access to user web bucket %s", hostname,bucket_name)
try:
apihost_url = apihost_util.get_user_static_url(runtime, hostname)
user_metadata.add_metadata("STATIC_CONTENT_URL",apihost_url)
if runtime == 'openshift':
return deploy_content_route_template(namespace,bucket_name, apihost_url)
else:
return deploy_content_ingress_template(namespace,bucket_name, apihost_url)
except Exception as e:
logging.error(e)
return False
def delete_ow_static_endpoint(ucfg):
"""
undeploy an ingress exposing a generic user web bucket
"""
namespace = ucfg.get("namespace")
runtime = cfg.get('nuvolaris.kube')
logging.info(f"*** removing static endpoint for {namespace}")
runtime = cfg.get('nuvolaris.kube')
ingress_class = util.get_ingress_class(runtime)
try:
res = ""
if(runtime=='openshift'):
route_name = static_route_name(namespace)
res = kube.kubectl("delete", "route",route_name)
return res
if(ingress_class == 'traefik'):
middleware_name = static_middleware_ingress_name(namespace)
res += kube.kubectl("delete", "middleware.traefik.containo.us",middleware_name)
ingress_name = static_ingress_name(namespace)
res += kube.kubectl("delete", "ingress",ingress_name)
return res
except Exception as e:
logging.warn(e)
return False
def delete_by_owner():
spec = kus.build("nginx-static")
res = kube.delete(spec)
logging.info(f"delete nuvolaris nginx static provider: {res}")
return res
def delete_by_spec():
spec = cfg.get("state.nginx-static.spec")
res = False
if spec:
res = kube.delete(spec)
logging.info(f"delete nuvolaris nginx static provider: {res}")
return res
def delete_nuv_ingresses():
logging.info("*** deleting nuvolaris static ingresses")
runtime = cfg.get('nuvolaris.kube')
ingress_class = util.get_ingress_class(runtime)
apihost_url = apihost_util.get_apihost(runtime)
hostname = apihost_util.extract_hostname(apihost_url)
should_delete_www = "www" not in hostname and runtime not in ["kind"]
try:
res = ""
if(runtime=='openshift'):
route_name = static_route_name("nuvolaris")
res = kube.kubectl("delete", "route",route_name)
if should_delete_www:
route_name = static_route_name("www-nuvolaris")
res += kube.kubectl("delete", "route",route_name)
return res
if(ingress_class == 'traefik'):
middleware_name = static_middleware_ingress_name("nuvolaris")
res += kube.kubectl("delete", "middleware.traefik.containo.us",middleware_name)
if should_delete_www:
middleware_name = static_middleware_ingress_name("www-nuvolaris")
res += kube.kubectl("delete", "middleware.traefik.containo.us",middleware_name)
ingress_name = static_ingress_name("nuvolaris")
res += kube.kubectl("delete", "ingress",ingress_name)
if should_delete_www:
ingress_name = static_ingress_name("www-nuvolaris")
res += kube.kubectl("delete", "ingress",ingress_name)
return res
except Exception as e:
logging.warn(e)
return False
def delete(owner=None):
delete_nuv_ingresses()
if owner:
return delete_by_owner()
else:
return delete_by_spec()
def patch(status, action, owner=None):
"""
Called by the operator patcher to create/delete minio static component
"""
try:
logging.info(f"*** handling request to {action} static")
if action == 'create':
msg = create(owner)
operator_util.patch_operator_status(status,'static','on')
else:
msg = delete(owner)
operator_util.patch_operator_status(status,'static','off')
logging.info(msg)
logging.info(f"*** hanlded request to {action} static")
except Exception as e:
logging.error('*** failed to update static: %s' % e)
operator_util.patch_operator_status(status,'static','error')