io.gravitee.node.license.LicenseService Maven / Gradle / Ivy
/**
* Copyright (C) 2015 The Gravitee team (http://gravitee.io)
*
* 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.
*/
package io.gravitee.node.license;
import ch.qos.logback.classic.Level;
import io.gravitee.common.service.AbstractService;
import io.gravitee.node.api.Node;
import io.gravitee.node.license.license3j.License3JLicense;
import io.gravitee.node.license.management.NodeLicenseManagementEndpoint;
import io.gravitee.node.management.http.endpoint.ManagementEndpointManager;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.IOException;
import java.nio.file.*;
import java.util.*;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import javax0.license3j.License;
import javax0.license3j.io.LicenseReader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.env.Environment;
import org.springframework.util.StringUtils;
/**
* @author David BRASSELY (david.brassely at graviteesource.com)
* @author GraviteeSource Team
*/
public class LicenseService extends AbstractService {
private final Logger logger = LoggerFactory.getLogger(LicenseService.class);
private static final String GRAVITEE_HOME_PROPERTY = "gravitee.home";
private static final String GRAVITEE_LICENSE_PROPERTY = "gravitee.license";
private static final String GRAVITEE_LICENSE_KEY = "license.key";
private static final String LICENSE_EXPIRE_AT = "expiryDate";
private static final long DAY_AS_LONG = 24 * 60 * 60 * 1000L;
@Autowired
private ManagementEndpointManager managementEndpointManager;
@Autowired
private Node node;
@Autowired
private Environment environment;
// prettier-ignore
private final byte [] key = new byte[] {
(byte)0x52,
(byte)0x53, (byte)0x41, (byte)0x00, (byte)0x30, (byte)0x82, (byte)0x01, (byte)0x22, (byte)0x30,
(byte)0x0D, (byte)0x06, (byte)0x09, (byte)0x2A, (byte)0x86, (byte)0x48, (byte)0x86, (byte)0xF7,
(byte)0x0D, (byte)0x01, (byte)0x01, (byte)0x01, (byte)0x05, (byte)0x00, (byte)0x03, (byte)0x82,
(byte)0x01, (byte)0x0F, (byte)0x00, (byte)0x30, (byte)0x82, (byte)0x01, (byte)0x0A, (byte)0x02,
(byte)0x82, (byte)0x01, (byte)0x01, (byte)0x00, (byte)0xD8, (byte)0x59, (byte)0xEC, (byte)0xB6,
(byte)0x27, (byte)0xF2, (byte)0x20, (byte)0x2F, (byte)0x7A, (byte)0x39, (byte)0x86, (byte)0x2B,
(byte)0x62, (byte)0xB6, (byte)0xEA, (byte)0x5B, (byte)0xE4, (byte)0x80, (byte)0xA0, (byte)0x32,
(byte)0x35, (byte)0xB3, (byte)0xC8, (byte)0xD5, (byte)0x4E, (byte)0xB7, (byte)0xA1, (byte)0xFE,
(byte)0x15, (byte)0x84, (byte)0xC7, (byte)0x75, (byte)0x66, (byte)0x8F, (byte)0x48, (byte)0xF1,
(byte)0xD6, (byte)0x30, (byte)0x7B, (byte)0x39, (byte)0xB7, (byte)0xD7, (byte)0x48, (byte)0x4F,
(byte)0xAF, (byte)0x38, (byte)0xC6, (byte)0xB9, (byte)0xBC, (byte)0x3C, (byte)0xEB, (byte)0xED,
(byte)0xB3, (byte)0x03, (byte)0xEE, (byte)0x1B, (byte)0x9D, (byte)0x85, (byte)0x8B, (byte)0xFC,
(byte)0x93, (byte)0x56, (byte)0x5C, (byte)0x09, (byte)0x91, (byte)0x41, (byte)0x22, (byte)0xE7,
(byte)0x4C, (byte)0xF6, (byte)0x94, (byte)0x9D, (byte)0xC5, (byte)0x71, (byte)0x3C, (byte)0x2D,
(byte)0xE4, (byte)0x4C, (byte)0x0E, (byte)0xD5, (byte)0x3D, (byte)0x6F, (byte)0xD7, (byte)0x20,
(byte)0x6C, (byte)0xD8, (byte)0xDD, (byte)0x12, (byte)0x4D, (byte)0xEA, (byte)0x55, (byte)0xDD,
(byte)0x26, (byte)0xA3, (byte)0x25, (byte)0xE3, (byte)0x83, (byte)0x9C, (byte)0x92, (byte)0x15,
(byte)0xC6, (byte)0x45, (byte)0xFE, (byte)0x9A, (byte)0x0F, (byte)0x47, (byte)0x86, (byte)0x45,
(byte)0x04, (byte)0xB6, (byte)0x44, (byte)0xFC, (byte)0x01, (byte)0x86, (byte)0x2A, (byte)0xF6,
(byte)0xE2, (byte)0xFD, (byte)0x37, (byte)0xF1, (byte)0xBB, (byte)0x70, (byte)0xAD, (byte)0x15,
(byte)0xE0, (byte)0x7C, (byte)0xB3, (byte)0x94, (byte)0xDE, (byte)0x2F, (byte)0xD2, (byte)0xC4,
(byte)0x4F, (byte)0xCB, (byte)0xA7, (byte)0x31, (byte)0x87, (byte)0x66, (byte)0xD0, (byte)0xF7,
(byte)0x5D, (byte)0x09, (byte)0x84, (byte)0x4B, (byte)0xB7, (byte)0x4B, (byte)0xE6, (byte)0x1C,
(byte)0xA7, (byte)0xD9, (byte)0xBE, (byte)0x9E, (byte)0xAD, (byte)0xE7, (byte)0x03, (byte)0xCC,
(byte)0xAB, (byte)0x04, (byte)0x4D, (byte)0xDF, (byte)0x92, (byte)0x8E, (byte)0xC5, (byte)0xA1,
(byte)0x04, (byte)0xD0, (byte)0x7F, (byte)0x89, (byte)0x71, (byte)0x2D, (byte)0x6D, (byte)0x8F,
(byte)0xCC, (byte)0x1E, (byte)0x25, (byte)0x5E, (byte)0x66, (byte)0xBF, (byte)0xA9, (byte)0xF8,
(byte)0x8C, (byte)0xEF, (byte)0x8E, (byte)0x6A, (byte)0xFA, (byte)0xCE, (byte)0xFA, (byte)0x79,
(byte)0xA0, (byte)0xDE, (byte)0x72, (byte)0xCB, (byte)0xBC, (byte)0x90, (byte)0x8E, (byte)0x29,
(byte)0x4C, (byte)0x40, (byte)0x05, (byte)0x61, (byte)0x5A, (byte)0x44, (byte)0x0D, (byte)0xC7,
(byte)0x46, (byte)0x65, (byte)0xA4, (byte)0x2A, (byte)0xD3, (byte)0xAA, (byte)0xEC, (byte)0x83,
(byte)0x78, (byte)0x52, (byte)0x04, (byte)0x2B, (byte)0xB4, (byte)0x15, (byte)0xE3, (byte)0x66,
(byte)0xD3, (byte)0xB5, (byte)0xE5, (byte)0xE7, (byte)0x04, (byte)0xB7, (byte)0xDB, (byte)0xFC,
(byte)0x46, (byte)0x32, (byte)0xC5, (byte)0x71, (byte)0x93, (byte)0xDD, (byte)0xE3, (byte)0x07,
(byte)0xD0, (byte)0xCD, (byte)0xD8, (byte)0x0E, (byte)0xDE, (byte)0xC1, (byte)0xA8, (byte)0xA6,
(byte)0x0F, (byte)0x45, (byte)0x8F, (byte)0x06, (byte)0x51, (byte)0xFE, (byte)0x72, (byte)0xB7,
(byte)0x61, (byte)0xCA, (byte)0x29, (byte)0x67, (byte)0x02, (byte)0x03, (byte)0x01, (byte)0x00,
(byte)0x01,
};
private LicenseReader reader;
private License license;
private Timer checkerTimer;
private LicenseChecker checkerTask;
private LicenseWatcher licenseWatcher;
public io.gravitee.node.api.license.License getLicense() {
return this.license == null ? null : new License3JLicense(this.license);
}
@Override
protected void doStart() throws Exception {
super.doStart();
// Ensure log level for license module to INFO
((ch.qos.logback.classic.Logger) logger).setLevel(Level.INFO);
this.loadLicense();
this.startLicenseChecker();
this.startLicenseWatcher();
managementEndpointManager.register(new NodeLicenseManagementEndpoint(node));
}
private void startLicenseChecker() {
checkerTimer = new Timer("gravitee-license-checker");
Calendar c = Calendar.getInstance();
c.setTime(new Date());
c.add(Calendar.DATE, 1);
c.set(Calendar.HOUR, 0);
c.set(Calendar.MINUTE, 0);
c.set(Calendar.SECOND, 0);
c.set(Calendar.MILLISECOND, 0);
checkerTask = new LicenseChecker();
checkerTimer.schedule(checkerTask, c.getTime(), DAY_AS_LONG);
}
private void startLicenseWatcher() {
String licenseKey = environment.getProperty(GRAVITEE_LICENSE_KEY);
if (!StringUtils.hasLength(licenseKey)) {
licenseWatcher = new LicenseWatcher(new File(getLicenseFile()));
licenseWatcher.setName("gravitee-license-watcher");
licenseWatcher.start();
}
}
@Override
protected void doStop() throws Exception {
super.doStop();
if (checkerTimer != null) {
checkerTimer.cancel();
}
if (licenseWatcher != null) {
licenseWatcher.close();
}
if (reader != null) {
reader.close();
}
}
@Override
protected String name() {
return "License service";
}
public void loadLicense() throws Exception {
byte[] licenseContent = getLicenseContent();
if (licenseContent.length > 0) {
try {
reader = new LicenseReader(new ByteArrayInputStream(licenseContent));
license = reader.read();
this.printLicenseInfo();
this.verify();
} catch (IllegalArgumentException iae) {
logger.error("License file is not valid", iae);
} catch (IOException ioe) {
logger.error("License file can not be read", ioe);
}
}
}
private byte[] getLicenseContent() {
byte[] licenseContent;
String licenseKey = environment.getProperty(GRAVITEE_LICENSE_KEY);
if (StringUtils.hasLength(licenseKey)) {
try {
return Base64.getDecoder().decode(licenseKey);
} catch (Exception ex) {
logger.error("Can't decode the license key.", ex);
return new byte[0];
}
}
try {
licenseContent = Files.readAllBytes(Paths.get(getLicenseFile()));
} catch (IOException e) {
logger.info("No license file found. Some plugins may be disabled");
return new byte[0];
}
return licenseContent;
}
private String getLicenseFile() {
String licenseFile = System.getProperty(GRAVITEE_LICENSE_PROPERTY);
if (licenseFile == null || licenseFile.isEmpty()) {
licenseFile = System.getProperty(GRAVITEE_HOME_PROPERTY) + File.separator + "license" + File.separator + GRAVITEE_LICENSE_KEY;
}
return licenseFile;
}
private void verify() throws Exception {
if (license == null) {
logger.debug("License will not be verified as it has not been loaded");
return;
}
boolean valid = license.isOK(key);
if (!valid) {
logger.error("License is not valid. Please contact GraviteeSource to ask for a valid license.");
stopNode();
}
if (license.isExpired()) {
logger.error("License is expired. Please contact GraviteeSource to ask for a renewed license.");
stopNode();
}
Date expiration = license.get(LICENSE_EXPIRE_AT).getDate();
long remainingDays = Math.round((expiration.getTime() - System.currentTimeMillis()) / (double) 86400000);
if (remainingDays <= 30) {
logger.warn("License will be no longer valid in {} days. Please contact GraviteeSource to renew it.", remainingDays);
}
}
private void printLicenseInfo() {
StringBuilder sb = new StringBuilder();
sb.append("License information: \n");
license
.getFeatures()
.forEach((name, feature) -> sb.append("\t").append(name).append(": ").append(feature.valueString()).append("\n"));
logger.info(sb.toString());
}
private class LicenseChecker extends TimerTask {
@Override
public void run() {
try {
LicenseService.this.verify();
} catch (Exception e) {
logger.error("An error occurred while checking license", e);
}
}
}
private class LicenseWatcher extends Thread {
private final File file;
private final AtomicBoolean stop = new AtomicBoolean(false);
LicenseWatcher(File file) {
this.file = file;
}
@Override
public void run() {
logger.debug("Watching license for next changes: {}", file.getAbsolutePath());
try (WatchService watcher = FileSystems.getDefault().newWatchService()) {
Path path = file.toPath().getParent();
path.register(watcher, StandardWatchEventKinds.ENTRY_MODIFY);
while (!stop.get()) {
WatchKey watchKey;
try {
watchKey = watcher.poll(25, TimeUnit.MILLISECONDS);
} catch (InterruptedException e) {
return;
}
if (watchKey == null) {
Thread.yield();
continue;
}
for (WatchEvent> event : watchKey.pollEvents()) {
WatchEvent.Kind> kind = event.kind();
@SuppressWarnings("unchecked")
WatchEvent ev = (WatchEvent) event;
Path filename = ev.context();
if (kind == StandardWatchEventKinds.OVERFLOW) {
Thread.yield();
continue;
} else if (kind == StandardWatchEventKinds.ENTRY_MODIFY && filename.toString().equals(file.getName())) {
LicenseService.this.loadLicense();
}
boolean valid = watchKey.reset();
if (!valid) {
break;
}
}
Thread.yield();
}
} catch (Exception e) {
logger.debug("An error occurred while watching license file", e);
}
}
void close() {
stop.set(true);
}
}
private void stopNode() throws Exception {
node.stop();
System.exit(0);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy