package com.thinkbiganalytics.server; /*- * #%L * thinkbig-service-app * %% * Copyright (C) 2017 ThinkBig Analytics * %% * Licensed 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. * #L% */ import com.thinkbiganalytics.KyloVersion; import com.thinkbiganalytics.KyloVersionUtil; import com.thinkbiganalytics.metadata.config.OperationalMetadataConfig; import com.thinkbiganalytics.metadata.upgrade.KyloUpgrader; import com.thinkbiganalytics.metadata.upgrade.UpgradeKyloConfig; import com.thinkbiganalytics.rest.SpringJerseyConfiguration; import org.apache.commons.lang3.StringUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.boot.autoconfigure.velocity.VelocityAutoConfiguration; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.cloud.config.server.EnableConfigServer; import org.springframework.context.ConfigurableApplicationContext; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Import; import org.springframework.context.support.ClassPathXmlApplicationContext; import org.springframework.scheduling.annotation.SchedulingConfigurer; import org.springframework.scheduling.config.ScheduledTaskRegistrar; import java.util.concurrent.Executor; import java.util.concurrent.Executors; @Configuration @SpringBootApplication @EnableAutoConfiguration(exclude = {VelocityAutoConfiguration.class}) @EnableConfigurationProperties @EnableConfigServer @Import({DatabaseConfiguration.class, OperationalMetadataConfig.class, SpringJerseyConfiguration.class}) @ComponentScan("com.thinkbiganalytics") public class KyloServerApplication implements SchedulingConfigurer { private static final Logger log = LoggerFactory.getLogger(KyloServerApplication.class); public static void main(String[] args) { KyloVersion dbVersion = getDatabaseVersion(); boolean skipUpgrade = KyloVersionUtil.isUpToDate(dbVersion); if (!skipUpgrade) { boolean upgradeComplete = false; do { log.info("Upgrading..."); System.setProperty(SpringApplication.BANNER_LOCATION_PROPERTY, "upgrade-banner.txt"); ConfigurableApplicationContext cxt = SpringApplication.run(UpgradeKyloConfig.class); KyloUpgrader upgrader = cxt.getBean(KyloUpgrader.class); upgradeComplete = upgrader.upgrade(); cxt.close(); } while (!upgradeComplete); log.info("Upgrading complete"); } else { log.info("Kylo v{} is up to date. Starting the application.",dbVersion); } System.setProperty(SpringApplication.BANNER_LOCATION_PROPERTY, "banner.txt"); SpringApplication.run("classpath:application-context.xml", args); } @Bean(destroyMethod = "shutdown") public Executor scheduledTaskExecutor() { return Executors.newScheduledThreadPool(25); } @Override public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) { scheduledTaskRegistrar.setScheduler(scheduledTaskExecutor()); } /** * Return the database version for Kylo. * * @return the version of Kylo stored in the database */ private static KyloVersion getDatabaseVersion() { try { //ensure native profile is there for spring to load String profiles = System.getProperty("spring.profiles.active", ""); if (!profiles.contains("native")) { profiles = StringUtils.isEmpty(profiles) ? "native" : profiles + ",native"; System.setProperty("spring.profiles.active", profiles); } //Spring is needed to load the Spring Cloud context so we can decrypt the passwords for the database connection ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("kylo-upgrade-check-application-context.xml"); ctx.refresh(); KyloUpgradeDatabaseVersionChecker upgradeDatabaseVersionChecker = (KyloUpgradeDatabaseVersionChecker) ctx.getBean("kyloUpgradeDatabaseVersionChecker"); KyloVersion kyloVersion = upgradeDatabaseVersionChecker.getDatabaseVersion(); ctx.close(); return kyloVersion; } catch (Exception e) { log.error("Failed get the database version prior to upgrade. The Kylo Upgrade application will load by default. {}", e.getMessage()); } return null; } }