2 * ========================LICENSE_START=================================
3 * Copyright (C) 2021 Nordix Foundation. All rights reserved.
4 * ======================================================================
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
9 * http://www.apache.org/licenses/LICENSE-2.0
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 * ========================LICENSE_END===================================
19 package org.onap.policy.clamp.controlloop.participant.kubernetes.service;
22 import java.io.FileOutputStream;
23 import java.io.IOException;
24 import java.io.PrintStream;
25 import java.lang.invoke.MethodHandles;
26 import java.nio.file.FileVisitResult;
27 import java.nio.file.Files;
28 import java.nio.file.Path;
29 import java.nio.file.Paths;
30 import java.nio.file.SimpleFileVisitor;
31 import java.nio.file.attribute.BasicFileAttributes;
32 import java.util.ArrayList;
33 import java.util.List;
35 import java.util.concurrent.ConcurrentHashMap;
36 import lombok.AccessLevel;
38 import org.onap.policy.clamp.controlloop.participant.kubernetes.exception.ServiceException;
39 import org.onap.policy.clamp.controlloop.participant.kubernetes.models.ChartInfo;
40 import org.onap.policy.clamp.controlloop.participant.kubernetes.parameters.ParticipantK8sParameters;
41 import org.onap.policy.common.utils.coder.CoderException;
42 import org.onap.policy.common.utils.coder.StandardCoder;
43 import org.slf4j.Logger;
44 import org.slf4j.LoggerFactory;
45 import org.springframework.stereotype.Component;
46 import org.springframework.util.FileSystemUtils;
47 import org.springframework.web.multipart.MultipartFile;
50 public class ChartStore {
51 private static final Logger LOGGER = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
53 private static final StandardCoder STANDARD_CODER = new StandardCoder();
55 private final ParticipantK8sParameters participantK8sParameters;
57 // ChartStore map contains chart name as key & ChartInfo as value.
58 @Getter(AccessLevel.PACKAGE)
59 private Map<String, ChartInfo> localChartMap = new ConcurrentHashMap<>();
64 public ChartStore(ParticipantK8sParameters participantK8sParameters) {
65 this.participantK8sParameters = participantK8sParameters;
66 this.restoreFromLocalFileSystem();
70 * Get local helm chart file.
72 * @param chart ChartInfo
73 * @return the chart file.
75 public File getHelmChartFile(ChartInfo chart) {
76 var appPath = getAppPath(chart.getChartName(), chart.getVersion());
77 return new File(appPath.toFile(), chart.getChartName());
81 * Get the override yaml file.
83 * @param chart ChartInfo
84 * @return the override yaml file
86 public File getOverrideFile(ChartInfo chart) {
87 var appPath = getAppPath(chart.getChartName(), chart.getVersion());
88 return new File(appPath.toFile(), "values.yaml");
93 * Saves the helm chart.
95 * @param chartInfo chartInfo
96 * @param chartFile helm chart file.
97 * @param overrideFile override file.
99 * @throws IOException incase of IO error
100 * @throws ServiceException incase of error.
102 public synchronized ChartInfo saveChart(ChartInfo chartInfo, MultipartFile chartFile, MultipartFile overrideFile)
103 throws IOException, ServiceException {
104 if (localChartMap.containsKey(key(chartInfo.getChartName(), chartInfo.getVersion()))) {
105 throw new ServiceException("Chart already exist");
107 var appPath = getAppPath(chartInfo.getChartName(), chartInfo.getVersion());
108 Files.createDirectories(appPath);
110 chartFile.transferTo(getHelmChartFile(chartInfo));
111 if (overrideFile != null) {
112 overrideFile.transferTo(getOverrideFile(chartInfo));
115 localChartMap.put(key(chartInfo), chartInfo);
116 storeChartInFile(chartInfo);
121 * Get the chart info.
123 * @param name name of the chart
124 * @param version version of the chart
127 public synchronized ChartInfo getChart(String name, String version) {
128 return localChartMap.get(key(name, version));
132 * Get all the charts installed.
134 * @return list of charts.
136 public synchronized List<ChartInfo> getAllCharts() {
137 return new ArrayList<>(localChartMap.values());
143 * @param chart chart info
145 public synchronized void deleteChart(ChartInfo chart) {
146 var appPath = getAppPath(chart.getChartName(), chart.getVersion());
148 FileSystemUtils.deleteRecursively(appPath);
149 } catch (IOException exc) {
150 LOGGER.warn("Could not delete chart from local file system : {}", appPath, exc);
153 localChartMap.remove(key(chart));
157 * Fetch the local chart directory of specific chart.
159 * @param chartName name of the chart
160 * @param chartVersion version of the chart
163 public Path getAppPath(String chartName, String chartVersion) {
164 return Path.of(participantK8sParameters.getLocalChartDirectory(), chartName, chartVersion);
167 private void storeChartInFile(ChartInfo chart) {
168 try (var out = new PrintStream(new FileOutputStream(getFile(chart)))) {
169 out.print(STANDARD_CODER.encode(chart));
170 } catch (Exception exc) {
171 LOGGER.warn("Could not store chart: {} {}", chart.getChartName(), exc);
175 private File getFile(ChartInfo chart) {
176 var appPath = getAppPath(chart.getChartName(), chart.getVersion()).toString();
177 return Path.of(appPath, participantK8sParameters.getInfoFileName()).toFile();
180 private synchronized void restoreFromLocalFileSystem() {
182 Path localChartDirectoryPath = Paths.get(participantK8sParameters.getLocalChartDirectory());
183 Files.createDirectories(localChartDirectoryPath);
184 restoreFromLocalFileSystem(localChartDirectoryPath);
185 } catch (Exception ioe) {
186 LOGGER.warn("Could not restore charts from local file system: {}", ioe);
190 private synchronized void restoreFromLocalFileSystem(Path localChartDirectoryPath)
193 Files.walkFileTree(localChartDirectoryPath, new SimpleFileVisitor<Path>() {
195 public FileVisitResult visitFile(Path localChartFile, BasicFileAttributes attrs) throws IOException {
197 // Decode only the json file excluding the helm charts
198 if (localChartFile.endsWith(participantK8sParameters.getInfoFileName())) {
199 ChartInfo chart = STANDARD_CODER.decode(localChartFile.toFile(), ChartInfo.class);
200 localChartMap.put(key(chart), chart);
202 return FileVisitResult.CONTINUE;
203 } catch (CoderException ce) {
204 throw new IOException("Error decoding chart file", ce);
210 private String key(ChartInfo chart) {
211 return key(chart.getChartName(), chart.getVersion());
214 private String key(String chartName, String chartVersion) {
215 return chartName + "_" + chartVersion;