All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.sitewhere.microservice.multitenant.MicroserviceTenantEngine Maven / Gradle / Ivy

There is a newer version: 3.0.13
Show newest version
/*
 * Copyright (c) SiteWhere, LLC. All rights reserved. http://www.sitewhere.com
 *
 * The software in this package is published under the terms of the CPAL v1.0
 * license, a copy of which has been included with this distribution in the
 * LICENSE.txt file.
 */
package com.sitewhere.microservice.multitenant;

import java.util.HashMap;
import java.util.Map;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.JsonNode;
import com.google.common.base.CaseFormat;
import com.google.inject.CreationException;
import com.google.inject.Injector;
import com.sitewhere.microservice.configuration.model.instance.PersistenceConfigurations;
import com.sitewhere.microservice.exception.ConcurrentK8sUpdateException;
import com.sitewhere.microservice.lifecycle.CompositeLifecycleStep;
import com.sitewhere.microservice.lifecycle.TenantEngineLifecycleComponent;
import com.sitewhere.microservice.scripting.Binding;
import com.sitewhere.microservice.scripting.ScriptManager;
import com.sitewhere.microservice.util.MarshalUtils;
import com.sitewhere.spi.SiteWhereException;
import com.sitewhere.spi.microservice.IFunctionIdentifier;
import com.sitewhere.spi.microservice.configuration.IConfigurableMicroservice;
import com.sitewhere.spi.microservice.lifecycle.ICompositeLifecycleStep;
import com.sitewhere.spi.microservice.lifecycle.ILifecycleProgressMonitor;
import com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine;
import com.sitewhere.spi.microservice.multitenant.ITenantEngineBootstrapper;
import com.sitewhere.spi.microservice.multitenant.ITenantEngineConfiguration;
import com.sitewhere.spi.microservice.multitenant.ITenantEngineSpecUpdateOperation;
import com.sitewhere.spi.microservice.multitenant.ITenantEngineStatusUpdateOperation;
import com.sitewhere.spi.microservice.scripting.IScriptManager;

import io.fabric8.kubernetes.client.utils.URLUtils;
import io.sitewhere.k8s.crd.ApiConstants;
import io.sitewhere.k8s.crd.ResourceLabels;
import io.sitewhere.k8s.crd.common.BootstrapState;
import io.sitewhere.k8s.crd.tenant.SiteWhereTenant;
import io.sitewhere.k8s.crd.tenant.configuration.TenantConfigurationTemplate;
import io.sitewhere.k8s.crd.tenant.dataset.TenantDatasetTemplate;
import io.sitewhere.k8s.crd.tenant.engine.SiteWhereTenantEngine;
import io.sitewhere.k8s.crd.tenant.engine.SiteWhereTenantEngineList;
import io.sitewhere.k8s.crd.tenant.engine.configuration.TenantEngineConfigurationTemplate;
import io.sitewhere.k8s.crd.tenant.engine.dataset.TenantEngineDatasetTemplate;
import okhttp3.MediaType;
import okhttp3.Request;
import okhttp3.RequestBody;
import okhttp3.Response;

/**
 * Component within microservice which runs for a specific tenant. Each tenant
 * engine has its own configuration and data model.
 * 
 * @param 
 */
public abstract class MicroserviceTenantEngine
	extends TenantEngineLifecycleComponent implements IMicroserviceTenantEngine {

    /** Number of milliseconds to wait between dataset bootstrap checks */
    private static final int DATASET_BOOTSTRAP_CHECK_INTERVAL = 5000;

    /** Tenant resource */
    private SiteWhereTenant tenantResource;

    /** Tenant engine resource */
    private SiteWhereTenantEngine tenantEngineResource;

    /** Active configuration */
    private T activeConfiguration;

    /** Guice injector containing configured components */
    private Injector injector;

    /** Script manager */
    private IScriptManager scriptManager = new ScriptManager();

    /** Tenant engine bootstrapper */
    private TenantEngineBootstrapper tenantEngineBootstrapper = new TenantEngineBootstrapper();

    public MicroserviceTenantEngine(SiteWhereTenantEngine tenantEngineResource) {
	this.tenantEngineResource = tenantEngineResource;
    }

    /*
     * @see
     * com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#getName(
     * )
     */
    @Override
    public String getName() {
	return getTenantEngineResource().getMetadata().getName();
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * getTenantResource()
     */
    @Override
    public SiteWhereTenant getTenantResource() {
	return tenantResource;
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * getTenantEngineResource()
     */
    @Override
    public SiteWhereTenantEngine getTenantEngineResource() {
	return tenantEngineResource;
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * getActiveConfiguration()
     */
    @Override
    public T getActiveConfiguration() {
	return activeConfiguration;
    }

    /*
     * @see com.sitewhere.microservice.lifecycle.TenantEngineLifecycleComponent#
     * getTenantEngine()
     */
    @Override
    public IMicroserviceTenantEngine getTenantEngine() {
	return this;
    }

    /*
     * (non-Javadoc)
     * 
     * @see com.sitewhere.server.lifecycle.LifecycleComponent#initialize(com.
     * sitewhere.spi.server.lifecycle.ILifecycleProgressMonitor)
     */
    @Override
    public void initialize(ILifecycleProgressMonitor monitor) throws SiteWhereException {
	resolveTenantResource();

	// Refresh active configuration from k8s resource.
	refreshConfiguration();

	// Load tenant engine components from Guice injector.
	loadEngineComponents();

	// Create step that will initialize components.
	ICompositeLifecycleStep init = new CompositeLifecycleStep("Initialize tenant engine " + getName());

	// Initialize script manager.
	init.addInitializeStep(this, getScriptManager(), true);

	// Initialize engine bootstrapper.
	init.addInitializeStep(this, getTenantEngineBootstrapper(), true);

	// Execute initialization steps.
	init.execute(monitor);

	// Allow subclass to execute initialization logic.
	tenantInitialize(monitor);
    }

    /**
     * Resolve the tenant resource references by tenant engine label.
     * 
     * @throws SiteWhereException
     */
    protected void resolveTenantResource() throws SiteWhereException {
	String tenantToken = getTenantEngineResource().getMetadata().getLabels()
		.get(ResourceLabels.LABEL_SITEWHERE_TENANT);
	if (tenantToken == null) {
	    throw new SiteWhereException("Tenant engine does not have a tenant label. Unable to resolve.");
	}
	String namespace = getMicroservice().getInstanceSettings().getKubernetesNamespace();
	SiteWhereTenant tenant = getMicroservice().getSiteWhereKubernetesClient().getTenants().inNamespace(namespace)
		.withName(tenantToken).get();
	if (tenant == null) {
	    throw new SiteWhereException(
		    String.format("Tenant engine label references a tenant '%s' which does not exist.", tenantToken));
	}
	this.tenantResource = tenant;
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * loadTenantEngineResource()
     */
    @Override
    public SiteWhereTenantEngine loadTenantEngineResource() throws SiteWhereException {
	String engineName = getTenantEngineResource().getMetadata().getName();
	String namespace = getMicroservice().getInstanceSettings().getKubernetesNamespace();
	SiteWhereTenantEngine found = getMicroservice().getSiteWhereKubernetesClient().getTenantEngines()
		.inNamespace(namespace).withName(engineName).get();
	if (found == null) {
	    throw new SiteWhereException(String.format("No tenant engine resource found with name '%s'.", engineName));
	}
	return found;
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * updateTenantEngineResource(io.sitewhere.k8s.crd.tenant.engine.
     * SiteWhereTenantEngine)
     */
    @Override
    public SiteWhereTenantEngine updateTenantEngineResource(SiteWhereTenantEngine engine) throws SiteWhereException {
	String engineName = getTenantEngineResource().getMetadata().getName();
	String namespace = getMicroservice().getInstanceSettings().getKubernetesNamespace();
	return getMicroservice().getSiteWhereKubernetesClient().getTenantEngines().inNamespace(namespace)
		.withName(engineName).createOrReplace(engine);
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * updateTenantEngineStatus(io.sitewhere.k8s.crd.tenant.engine.
     * SiteWhereTenantEngine)
     */
    @Override
    public SiteWhereTenantEngine updateTenantEngineStatus(SiteWhereTenantEngine engine) throws SiteWhereException {
	try {
	    final String statusUri = URLUtils.join(getMicroservice().getKubernetesClient().getMasterUrl().toString(),
		    "apis", ApiConstants.SITEWHERE_API_GROUP, ApiConstants.SITEWHERE_API_VERSION, "namespaces",
		    engine.getMetadata().getNamespace(), ApiConstants.SITEWHERE_TENANT_ENGINE_CRD_PLURAL,
		    engine.getMetadata().getName(), "status");
	    final RequestBody requestBody = RequestBody.create(MediaType.parse("application/json"),
		    MarshalUtils.marshalJson(engine));
	    Response response = getMicroservice().getKubernetesClient().getHttpClient()
		    .newCall(new Request.Builder().method("PUT", requestBody).url(statusUri).build()).execute();
	    byte[] content = response.body().bytes();
	    response.close();
	    JsonNode json = MarshalUtils.marshalJsonNode(content);
	    SiteWhereTenantEngine result = MarshalUtils.unmarshalJsonNode(json, SiteWhereTenantEngine.class);
	    return result;
	} catch (JsonProcessingException e) {
	    throw new ConcurrentK8sUpdateException("Tenant engine status update failed due to conflict.", e);
	} catch (Throwable e) {
	    throw new SiteWhereException("Unhandled exception updating tenant engine status.", e);
	}
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * executeTenantEngineSpecUpdate(com.sitewhere.spi.microservice.multitenant.
     * ITenantEngineSpecUpdateOperation)
     */
    @Override
    public SiteWhereTenantEngine executeTenantEngineSpecUpdate(ITenantEngineSpecUpdateOperation operation)
	    throws SiteWhereException {
	return operation.execute(this);
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * executeTenantEngineStatusUpdate(com.sitewhere.spi.microservice.multitenant.
     * ITenantEngineStatusUpdateOperation)
     */
    @Override
    public SiteWhereTenantEngine executeTenantEngineStatusUpdate(ITenantEngineStatusUpdateOperation operation)
	    throws SiteWhereException {
	return operation.execute(this);
    }

    /**
     * Parse tenant engine configuraion into expected configuration type.
     * 
     * @return
     * @throws SiteWhereException
     */
    protected T parseConfiguration() throws SiteWhereException {
	try {
	    JsonNode configuration = getTenantEngineResource().getSpec().getConfiguration();
	    return MarshalUtils.unmarshalJsonNode(configuration, getConfigurationClass());
	} catch (JsonProcessingException e) {
	    throw new SiteWhereException("Unable to parse tenant engine configuration.", e);
	} catch (Throwable t) {
	    throw new SiteWhereException("Unhandled exception parsing tenant engine configuration.", t);
	}
    }

    /**
     * Refresh active configuration from k8s resource.
     * 
     * @throws SiteWhereException
     */
    protected void refreshConfiguration() throws SiteWhereException {
	this.activeConfiguration = parseConfiguration();
	if (getLogger().isDebugEnabled()) {
	    getLogger().debug(String.format("Refreshed tenant engine configuration: \n%s\n\n",
		    MarshalUtils.marshalJsonAsPrettyString(getActiveConfiguration())));
	}
	try {
	    // Inherit existing bindings from the microsevice injector.
	    this.injector = ((IConfigurableMicroservice) getMicroservice()).getInjector()
		    .createChildInjector(createConfigurationModule());
	    PersistenceConfigurations configs = getInjector().getInstance(PersistenceConfigurations.class);
	    if (configs == null) {
		getLogger().debug("Did not find persistence configurations.");
	    } else {
		getLogger().debug(String.format("Found persistence configs:\n%s\n\n",
			MarshalUtils.marshalJsonAsPrettyString(configs)));
	    }
	} catch (CreationException e) {
	    throw new SiteWhereException("Guice configuration module failed to initialize.", e);
	}
    }

    /*
     * (non-Javadoc)
     * 
     * @see
     * com.sitewhere.server.lifecycle.LifecycleComponent#start(com.sitewhere.spi
     * .server.lifecycle.ILifecycleProgressMonitor)
     */
    @Override
    public void start(ILifecycleProgressMonitor monitor) throws SiteWhereException {

	// Create step that will start components.
	ICompositeLifecycleStep start = new CompositeLifecycleStep("Start tenant engine " + getName());

	// Start tenant script manager.
	start.addStartStep(this, getScriptManager(), true);

	// Execute startup steps.
	start.execute(monitor);

	// Allow subclass to execute startup logic.
	tenantStart(monitor);

	// Execute bootstrap in background
	startNestedComponent(getTenantEngineBootstrapper(), monitor, true);
    }

    /*
     * (non-Javadoc)
     * 
     * @see
     * com.sitewhere.server.lifecycle.LifecycleComponent#stop(com.sitewhere.spi.
     * server.lifecycle.ILifecycleProgressMonitor)
     */
    @Override
    public void stop(ILifecycleProgressMonitor monitor) throws SiteWhereException {

	// Stop the engine bootstrapper.
	stopNestedComponent(getTenantEngineBootstrapper(), monitor);

	// Allow subclass to execute shutdown logic.
	tenantStop(monitor);

	// Create step that will stop components.
	ICompositeLifecycleStep stop = new CompositeLifecycleStep("Stop tenant engine " + getName());

	// Stop tenant script manager.
	stop.addStopStep(this, getScriptManager());

	// Execute shutdown steps.
	stop.execute(monitor);
    }

    /*
     * (non-Javadoc)
     * 
     * @see
     * com.sitewhere.server.lifecycle.LifecycleComponent#terminate(com.sitewhere
     * .spi.server.lifecycle.ILifecycleProgressMonitor)
     */
    @Override
    public void terminate(ILifecycleProgressMonitor monitor) throws SiteWhereException {
	// Create step that will terminate components.
	ICompositeLifecycleStep stop = new CompositeLifecycleStep("Terminate tenant engine " + getName());

	// Terminate tenant script manager.
	stop.addTerminateStep(this, getScriptManager());

	// Execute terminate steps.
	stop.execute(monitor);
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * loadEngineComponents()
     */
    @Override
    public void loadEngineComponents() throws SiteWhereException {
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * getInjector()
     */
    @Override
    public Injector getInjector() {
	return injector;
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * getConfigurationTemplate()
     */
    @Override
    public TenantEngineConfigurationTemplate getConfigurationTemplate() throws SiteWhereException {
	String tctName = getTenantResource().getSpec().getConfigurationTemplate();
	if (tctName == null) {
	    throw new SiteWhereException("No tenant configuration template specified.");
	}
	TenantConfigurationTemplate tct = getMicroservice().getSiteWhereKubernetesClient()
		.getTenantConfigurationTemplates().withName(tctName).get();
	if (tct == null) {
	    throw new SiteWhereException(String.format("No tenant configuration template found for '%s'.", tctName));
	}
	String functionName = CaseFormat.LOWER_HYPHEN.to(CaseFormat.LOWER_CAMEL,
		getMicroservice().getIdentifier().getPath());
	String tectName = tct.getSpec().getTenantEngineTemplates().get(functionName);
	if (tectName == null) {
	    throw new SiteWhereException(
		    String.format("No tenant engine configuration template listed for '%s'.", tectName));
	}
	TenantEngineConfigurationTemplate tect = getMicroservice().getSiteWhereKubernetesClient()
		.getTenantEngineConfigurationTemplates().withName(tectName).get();
	if (tect == null) {
	    throw new SiteWhereException(
		    String.format("No tenant engine configuration template found for '%s'.", tectName));
	}
	return tect;
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * getDatasetTemplate()
     */
    @Override
    public TenantEngineDatasetTemplate getDatasetTemplate() throws SiteWhereException {
	String tdtName = getTenantResource().getSpec().getDatasetTemplate();
	if (tdtName == null) {
	    throw new SiteWhereException("No tenant dataset template specified.");
	}
	TenantDatasetTemplate tdt = getMicroservice().getSiteWhereKubernetesClient().getTenantDatasetTemplates()
		.withName(tdtName).get();
	if (tdt == null) {
	    throw new SiteWhereException(String.format("No tenant dataset template found for '%s'.", tdtName));
	}
	String functionName = CaseFormat.LOWER_HYPHEN.to(CaseFormat.LOWER_CAMEL,
		getMicroservice().getIdentifier().getPath());
	String tedtName = tdt.getSpec().getTenantEngineTemplates().get(functionName);
	if (tedtName == null) {
	    // Not all tenant engines use datasets to bootstrap their data.
	    return null;
	}
	TenantEngineDatasetTemplate tedt = getMicroservice().getSiteWhereKubernetesClient()
		.getTenantEngineDatasetTemplates().withName(tedtName).get();
	if (tedt == null) {
	    throw new SiteWhereException(String.format("No tenant engine dataset template found for '%s'.", tedtName));
	}
	return tedt;
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * getTenantEngineBootstrapper()
     */
    @Override
    public ITenantEngineBootstrapper getTenantEngineBootstrapper() throws SiteWhereException {
	return tenantEngineBootstrapper;
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * getTenantBootstrapPrerequisites()
     */
    @Override
    public IFunctionIdentifier[] getTenantBootstrapPrerequisites() {
	return new IFunctionIdentifier[0];
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * setDatasetBootstrapBindings(com.sitewhere.microservice.scripting.Binding)
     */
    @Override
    public void setDatasetBootstrapBindings(Binding binding) throws SiteWhereException {
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * waitForTenantDatasetBootstrapped(com.sitewhere.spi.microservice.
     * IFunctionIdentifier)
     */
    @Override
    public void waitForTenantDatasetBootstrapped(IFunctionIdentifier identifier) throws SiteWhereException {
	while (true) {
	    Map labels = new HashMap<>();
	    labels.put(ResourceLabels.LABEL_SITEWHERE_TENANT, getTenantResource().getMetadata().getName());
	    labels.put(ResourceLabels.LABEL_SITEWHERE_FUNCTIONAL_AREA, identifier.getPath());
	    SiteWhereTenantEngineList matches = getMicroservice().getSiteWhereKubernetesClient().getTenantEngines()
		    .inNamespace(getTenantResource().getMetadata().getNamespace()).withLabels(labels).list();
	    SiteWhereTenantEngine match = null;
	    if (matches.getItems().size() == 1) {
		match = matches.getItems().get(0);
	    } else if (matches.getItems().size() > 1) {
		throw new SiteWhereException(String.format("Multiple tenant engines found for dependent dataset '%s'",
			identifier.getPath()));
	    }
	    if (match != null) {
		BootstrapState state = match.getStatus() != null ? match.getStatus().getBootstrapState() : null;
		if (state != null && state == BootstrapState.Bootstrapped) {
		    getLogger().info(String.format("Dataset for '%s' has been bootstrapped.", identifier.getPath()));
		    return;
		}
		getLogger()
			.info(String.format("Waiting for dataset for '%s' to become available. Current state is '%s'.",
				identifier.getPath(), state));
	    } else {
		getLogger()
			.info(String.format("Waiting for dataset for '%s' to become available.", identifier.getPath()));
	    }

	    try {
		Thread.sleep(DATASET_BOOTSTRAP_CHECK_INTERVAL);
	    } catch (InterruptedException e) {
		getLogger().info("Interrupted while waiting for dataset to become available.");
	    }
	}
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * onTenantBootstrapComplete()
     */
    @Override
    public void onTenantBootstrapComplete() {
    }

    /*
     * @see com.sitewhere.spi.microservice.multitenant.IMicroserviceTenantEngine#
     * getScriptManager()
     */
    @Override
    public IScriptManager getScriptManager() {
	return scriptManager;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy