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

reliza.java.client.Library Maven / Gradle / Ivy

The newest version!
package reliza.java.client;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.time.Instant;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Base64;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.stream.Collectors;

import org.apache.commons.collections4.CollectionUtils;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.StringUtils;
import lombok.extern.slf4j.Slf4j;
import okhttp3.OkHttpClient;

import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;

import retrofit2.Call;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.converter.jackson.JacksonConverterFactory;

import reliza.java.client.interceptors.BasicAuthInterceptor;
import reliza.java.client.interceptors.CsrfInterceptor;
import reliza.java.client.responses.FullRelease;
import reliza.java.client.responses.GraphQLResponse;
import reliza.java.client.responses.ProjectVersion;
import reliza.java.client.responses.ReleaseData;
import reliza.java.client.responses.ArtifactData.TagRecord;

/**
 * Defines a library which holds all the parameters from Flags and uses it to make API calls
 */
@Slf4j
public class Library {
	Flags flags;
	RHService rhs;
	ObjectMapper OM = new ObjectMapper();
	String RELEASE_GQL_DATA;
	String FULL_RELEASE_GQL_DATA;
	
	/**
	 * Initializing Flags and RHService to call API. Default endpoint is set to https://app.relizahub.com, however can be modified
	 * by setting baseUrl in flags builder.
	 * @param flags - Class Flags which contains all possible parameters, flags must be created using builder pattern.
	 */   
	public Library(Flags flags) {
		this.flags = flags;
		this.OM.registerModule(new JavaTimeModule());
		this.OM.setSerializationInclusion(JsonInclude.Include.NON_EMPTY);
		OkHttpClient client = new OkHttpClient.Builder()
				.addInterceptor(new BasicAuthInterceptor(flags.getApiKeyId(), flags.getApiKey()))
				.addInterceptor(new CsrfInterceptor(flags.getBaseUrl()))
				.build();
		Retrofit retrofit = new Retrofit.Builder()
				.baseUrl(flags.getBaseUrl())
				.addConverterFactory(JacksonConverterFactory.create(OM))
				.client(client)
				.build();
		this.rhs = retrofit.create(RHService.class);
		
		this.RELEASE_GQL_DATA = ""
			+ "uuid \n"
			+ "createdType \n"
			+ "lastUpdatedBy \n"
			+ "createdDate \n"
			+ "version \n"
			+ "status \n"
			+ "org \n"
			+ "project \n"
			+ "branch \n"
			+ "parentReleases { \n"
				+ "timeSent \n"
				+ "release \n"
				+ "artifact \n"
				+ "type \n"
				+ "namespace \n"
				+ "properties \n"
				+ "state \n"
				+ "replicas { \n"
					+ "id \n"
					+ "state \n"
				+ "} \n"
				+ "isInError \n"
			+ "} \n"
			+ "optionalReleases { \n"
				+ "timeSent \n"
				+ "release \n"
				+ "artifact \n"
				+ "type \n"
				+ "namespace \n"
				+ "properties \n"
				+ "state \n"
				+ "replicas { \n"
					+ "id \n"
					+ "state \n"
				+ "} \n"
				+ "isInError \n"
			+ "} \n"
			+ "sourceCodeEntry \n"
			+ "commits \n"
			+ "commitTime \n"
			+ "artifacts \n"
			+ "type \n"
			+ "notes \n"
			+ "approvals \n"
			+ "timing { \n"
				+ "lifecycle \n"
				+ "dateFrom \n"
				+ "dateTo \n"
				+ "instanceUuid \n"
				+ "environment \n"
				+ "event \n"
				+ "duration \n"
			+ "} \n"
			+ "decoratedVersionString \n"
			+ "endpoint";
		
		this.FULL_RELEASE_GQL_DATA = RELEASE_GQL_DATA + "\n"
			+ "sourceCodeEntryDetails { \n"
				+ "uuid \n"
				+ "branchUuid\n"
				+ "vcsUuid\n"
				+ "vcsBranch\n"
				+ "commit\n"
				+ "commits\n"
				+ "commitMessage\n"
				+ "vcsTag\n"
				+ "notes\n"
				+ "org\n"
				+ "dateActual \n"
			+ "} \n"
			+ "vcsRepository { \n"
				+ "uuid \n"
				+ "name \n"
				+ "org \n"
				+ "uri \n"
				+ "type \n"
			+ "} \n"
			+ "artifactDetails { \n"
				+ "uuid \n"
				+ "identifier \n"
				+ "org \n"
				+ "branch \n"
				+ "buildId \n"
				+ "buildUri \n"
				+ "cicdMeta \n"
				+ "digests \n"
				+ "isInternal \n"
				+ "artifactType { \n"
					+ "name \n"
					+ "aliases \n"
				+ "} \n"
				+ "notes \n"
				+ "tagRecords {\n"
					+ "key \n"
					+ "value \n"
				+ "} \n"
				+ "dateFrom \n"
				+ "dateTo \n"
				+ "buildDuration \n"
				+ "packageType \n"
				+ "version \n"
				+ "publisher \n"
				+ "group \n"
				+ "dependencies \n"
			+ "} \n"
			+ "projectName \n"
			+ "namespace";
	}
	
	/**
	 * Method that denotes we are obtaining the next available release version for the branch.
	 * Note that if the call succeeds, the version assignment will be recorded and will not be given again by Reliza Hub, even if it is not consumed. It will create the release in pending status. 

* Method itself does not require parameters but requires that the Flags class passed during library initialization contains these parameters.

* - apiKeyId (required) - flag for project api id or organization-wide read-write api id.
* - apiKey (required) - flag for project api key or organization-wide read-write api key.
* - branch (required) - flag to denote branch. If the branch is not recorded yet, Reliza Hub will attempt to create it.
* - projectId (optional) - flag to denote project uuid. Required if organization-wide read-write key is used, ignored if project specific api key is used.
* - versionSchema (optional for existing branches, required for new branches) - flag to denote branch versionSchema. If supplied for an existing branch and versionSchema is different from current, it will override current versionSchema.
* - vcsUri (optional) - flag to denote vcs uri. Currently this flag is needed if we want to set a commit for the release. However, soon it will be needed only if the vcs uri is not yet set for the project.
* - vcsType (optional) - flag to denote vcs type. Supported values: git, svn, mercurial. As with vcsuri, this flag is needed if we want to set a commit for the release. However, soon it will be needed only if the vcs uri is not yet set for the project.
* - commitHash (optional) - flag to denote vcs commit id or hash. This is needed to provide source code entry metadata into the release.
* commitMessage (optional) - flag to denote the commit message of the commit.
* commitList (optional) - flag to denote base64 encoded list of commits. Usually formatted by calling git log $START_COMMIT..$CURRENT_COMMIT --date=iso-strict --pretty="%H|||%ad|||%s" | base64 -w 0
* - dateActual (optional) - flag to denote date time with timezone when commit was made, iso strict formatting with timezone is required, i.e. for git use git log --date=iso-strict.
* - vcsTag (optional) - flag to denote vcs tag. This is needed to include vcs tag into commit, if present.
* - manual (optional) - flag to indicate a manual release. Sets status as "draft", otherwise "pending" status is used.
* - metadata (optional) - flag to set version metadata. This may be semver metadata or custom version schema metadata.
* - modifier (optional) - flag to set version modifier. This may be semver modifier or custom version schema.
* - action (optional) * - onlyVersion (optional) - boolean flag to skip creation of the release. Default is false. * @return returns ProjectVersion if successful API call and null otherwise. */ public ProjectVersion getVersion() { Map variables = new HashMap<>(); variables.put("branch", flags.getBranch()); variables.put("versionSchema", flags.getVersionSchema()); variables.put("project", flags.getProjectId()); variables.put("onlyVersion", flags.getOnlyVersion()); variables.put("modifier", flags.getModifier()); variables.put("metadata", flags.getMetadata()); variables.put("action", flags.getAction()); if (StringUtils.isNotEmpty(flags.getCommitHash()) || StringUtils.isNotEmpty(flags.getCommitMessage())) { Map commitMap = new HashMap<>(); commitMap.put("commit", flags.getCommitHash()); commitMap.put("commitMessage", flags.getCommitMessage()); commitMap.put("uri", flags.getVcsUri()); commitMap.put("type", flags.getVcsType()); commitMap.put("vcsTag", flags.getVcsTag()); commitMap.put("dateActual", flags.getDateActual()); variables.put("sourceCodeEntry", commitMap); } if (StringUtils.isNotEmpty(flags.getCommitList())) { String commits = new String(Base64.getDecoder().decode(flags.getCommitList())); List commitList = Arrays.asList(StringUtils.split(commits, System.lineSeparator())); List> commitsInBody = new ArrayList<>(); for (int i = 0; i < commitList.size(); i++) { Map singleCommit = new HashMap<>(); List commitParts = Arrays.asList(StringUtils.split(commitList.get(i), "|||")); singleCommit.put("commit", commitParts.get(0)); singleCommit.put("dateActual", commitParts.get(1)); singleCommit.put("commitMessage", commitParts.get(2)); if (commitParts.size() > 3) { singleCommit.put("commitAuthor", commitParts.get(3)); singleCommit.put("commitEmail", commitParts.get(4)); } commitsInBody.add(singleCommit); if (i == 0 && !StringUtils.isNotEmpty(flags.getCommitHash())) { Map commitMap = new HashMap<>(); commitMap.put("commit", commitParts.get(0)); commitMap.put("dateActual", commitParts.get(1)); commitMap.put("commitMessage", commitParts.get(2)); if (commitParts.size() > 3) { commitMap.put("commitAuthor", commitParts.get(3)); commitMap.put("commitEmail", commitParts.get(4)); } commitMap.put("vcsTag", flags.getVcsTag()); commitMap.put("uri", flags.getVcsUri()); commitMap.put("type", flags.getVcsType()); variables.put("sourceCodeEntry", commitMap); } } variables.put("commits", commitsInBody); } if (flags.getManual()) { variables.put("status", "DRAFT"); } String query = "" + "mutation ($GetNewVersionInput: GetNewVersionInput) { \n" + "getNewVersion(project:$GetNewVersionInput) \n" + "}"; Map input = new HashMap<>(); input.put("GetNewVersionInput", variables); Map body = new HashMap<>(); body.put("query", query); body.put("variables", input); Call call = rhs.getVersion(body); Map response = execute(call); return response == null ? null : OM.convertValue(response.get("getNewVersion"), ProjectVersion.class); } /** * Method that denotes we are sending Release Metadata of a Project to Reliza Hub. * Note that Reliza Hub will only allow you to send release data once per project version.

* Multiple artifacts per release are supported. In which case artifact specific flags (artid, artbuildid, artbuilduri, artcimeta, arttype, artdigests, tagkey and tagval must be repeated for each artifact).

* For sample of how to use workflow in CI, refer to the GitHub Actions build yaml of this project here * https://github.com/relizaio/reliza-cli/blob/master/.github/workflows/dockerimage.yml

* Method itself does not require parameters but requires that the Flags class passed during library initialization contains these parameters.

* - apiKeyId (required) - flag for project api id or organization-wide read-write api id.
* - apiKey (required) - flag for project api key or organization-wide read-write api key.
* - branch (required) - flag to denote branch. If branch is not recorded yet, Reliza Hub will attempt to create it.
* - version (required) - version. Note that Reliza Hub will reject the call if a release with this exact version is already present for this project.
* - endPoint (optional) - flag to denote test endpoint URI. This would be useful for systems where every release gets test URI.
* - projectId (optional) - flag to denote project uuid. Required if organization-wide read-write key is used, ignored if project specific api key is used.
* - vcsUri (optional) - flag to denote vcs uri. Currently this flag is needed if we want to set a commit for the release. However, soon it will be needed only if the vcs uri is not yet set for the project.
* - vcsType (optional) - flag to denote vcs type. Supported values: git, svn, mercurial. As with vcsuri, this flag is needed if we want to set a commit for the release. However, soon it will be needed only if the vcs uri is not yet set for the project.
* - commitHash (optional) - flag to denote vcs commit id or hash. This is needed to provide source code entry metadata into the release.
* commitMessage (optional) - flag to denote the commit message of the commit.
* commitList (optional) - flag to denote base64 encoded list of commits. Usually formatted by calling git log $START_COMMIT..$CURRENT_COMMIT --date=iso-strict --pretty="%H|||%ad|||%s" | base64 -w 0
* - dateActual (optional) - flag to denote date time with timezone when commit was made, iso strict formatting with timezone is required, i.e. for git use git log --date=iso-strict.
* - vcsTag (optional) - flag to denote vcs tag. This is needed to include vcs tag into commit, if present.
* - status (optional) - flag to denote release status. Supply "rejected" for failed releases, otherwise "completed" is used.
* - artId (optional) - flag to denote artifact identifier. This is required to add artifact metadata into release.
* - artBuildId (optional) - flag to denote artifact build id. This flag is optional and may be used to indicate build system id of the release (i.e., this could be circleci build number).
* - artBuildUri (optional) - flag to denote artifact build uri. This flag is optional and is used to denote the uri for where the build takes place.
* - artCiMeta (optional) - flag to denote artifact CI metadata. This flag is optional and like artbuildid may be used to indicate build system metadata in free form.
* - artType (optional) - flag to denote artifact type. This flag is used to denote artifact type. Types are based on CycloneDX spec. Supported values: Docker, File, Image, Font, Library, Application, Framework, OS, Device, Firmware.
* - dateStart (optional, if used there must be one datestart flag entry per artifact) - flag to denote artifact build start date and time, must conform to ISO strict date, i.e. "2021-01-12T19:43:32Z".
* - dateEnd (optional, if used there must be one datestart flag entry per artifact) - flag to denote artifact build end date and time, must conform to ISO strict date, i.e. "2021-01-12T19:43:32Z".
* - artPublisher (optional, if used there must be one publisher flag entry per artifact) - flag to denote artifact publisher.
* - artVersion (optional, if used there must be one version flag entry per artifact) - flag to denote artifact version if different from release version.
* - artPackage (optional, if used there must be one package flag entry per artifact) - flag to denote artifact package type according to CycloneDX spec: MAVEN, NPM, NUGET, GEM, PYPI, DOCKER.
* - artGroup (optional, if used there must be one group flag entry per artifact) - flag to denote artifact group.
* - artDigests (optional) - flag to denote artifact digests. This flag is used to indicate artifact digests. By convention, digests must be prefixed with type followed by colon and then actual digest hash, i.e.
* sha256:4e8b31b19ef16731a6f82410f9fb929da692aa97b71faeb1596c55fbf663dcdd - here type is sha256 and digest is 4e8b31b19ef16731a6f82410f9fb929da692aa97b71faeb1596c55fbf663dcdd. Multiple digests are supported and must be comma separated. i.e.
* sha256:4e8b31b19ef16731a6f82410f9fb929da692aa97b71faeb1596c55fbf663dcdd,sha1:fe4165996a41501715ea0662b6a906b55e34a2a1
* - tagKeys (optional, but every tag key must have corresponding tag value) - flag to denote keys of artifact tags. Multiple tag keys per artifact are supported and must be comma separated. I.e.:tagKeyArr(key1,key2)
* - tagVals (optional, but every tag value must have corresponding tag key) - flag to denote values of artifact tags. Multiple tag values per artifact are supported and must be comma separated. I.e.:tagValArr(val1,val2) * @return returns ReleaseData if successful API call and null otherwise. */ public ReleaseData addRelease() { Map variables = new HashMap<>(); variables.put("branch", flags.getBranch()); variables.put("version", flags.getVersion()); variables.put("status", StringUtils.isNotEmpty(flags.getStatus()) ? flags.getStatus().toUpperCase() : null); variables.put("endpoint", flags.getEndPoint()); variables.put("project", flags.getProjectId()); if (StringUtils.isNotEmpty(flags.getCommitHash())) { Map commitMap = new HashMap<>(); commitMap.put("commit", flags.getCommitHash()); commitMap.put("commitMessage", flags.getCommitMessage()); commitMap.put("uri", flags.getVcsUri()); commitMap.put("type", flags.getVcsType()); commitMap.put("vcsTag", flags.getVcsTag()); commitMap.put("dateActual", flags.getDateActual()); variables.put("sourceCodeEntry", commitMap); } if (StringUtils.isNotEmpty(flags.getCommitList())) { String commits = new String(Base64.getDecoder().decode(flags.getCommitList())); List commitList = Arrays.asList(StringUtils.split(commits, System.lineSeparator())); List> commitsInBody = new ArrayList<>(); for (int i = 0; i < commitList.size(); i++) { Map singleCommit = new HashMap<>(); List commitParts = Arrays.asList(StringUtils.split(commitList.get(i), "|||")); singleCommit.put("commit", commitParts.get(0)); singleCommit.put("dateActual", commitParts.get(1)); singleCommit.put("commitMessage", commitParts.get(2)); if (commitParts.size() > 3) { singleCommit.put("commitAuthor", commitParts.get(3)); singleCommit.put("commitEmail", commitParts.get(4)); } commitsInBody.add(singleCommit); if (i == 0 && !StringUtils.isNotEmpty(flags.getCommitHash())) { Map commitMap = new HashMap<>(); commitMap.put("commit", commitParts.get(0)); commitMap.put("dateActual", commitParts.get(1)); commitMap.put("commitMessage", commitParts.get(2)); if (commitParts.size() > 3) { commitMap.put("commitAuthor", commitParts.get(3)); commitMap.put("commitEmail", commitParts.get(4)); } commitMap.put("vcsTag", flags.getVcsTag()); commitMap.put("uri", flags.getVcsUri()); commitMap.put("type", flags.getVcsType()); variables.put("sourceCodeEntry", commitMap); } } variables.put("commits", commitsInBody); } if (CollectionUtils.isNotEmpty(flags.getArtId())) { List> artifacts = new ArrayList>(); for (String artId : flags.getArtId()) { if (StringUtils.isEmpty(artId)) { log.error("artid flag cannot be empty"); return null; } Map artifact = new HashMap<>(); artifact.put("identifier", artId); artifacts.add(artifact); } Map> artFlags = new HashMap>(); artFlags.put("buildId", flags.getArtBuildId()); artFlags.put("buildUri", flags.getArtBuildUri()); artFlags.put("cicdMeta", flags.getArtCiMeta()); artFlags.put("type", flags.getArtType()); artFlags.put("artifactVersion", flags.getArtVersion()); artFlags.put("publisher", flags.getArtPublisher().stream().map( s -> StringUtils.isNotEmpty(s) ? s.toUpperCase() : s).collect(Collectors.toList())); artFlags.put("packageType", flags.getArtPackage()); artFlags.put("group", flags.getArtGroup()); artFlags.put("dateFrom", flags.getDateStart()); artFlags.put("dateTo", flags.getDateEnd()); for (String key : artFlags.keySet()) { if (CollectionUtils.isNotEmpty(artFlags.get(key)) && artFlags.get(key).size() != flags.getArtId().size()) { log.error("number of art" + key.toLowerCase() + " flags must be either zero or match number of artid flags"); return null; } else if (CollectionUtils.isNotEmpty(artFlags.get(key))) { for (int j = 0; j < flags.getArtId().size(); j++) { if (StringUtils.isEmpty(artFlags.get(key).get(j))) { continue; } artifacts.get(j).put(key, artFlags.get(key).get(j)); } } } if (CollectionUtils.isNotEmpty(flags.getArtDigests()) && flags.getArtDigests().size() != flags.getArtId().size()) { log.error("number of artdigests flags must be either zero or match number of artid flags"); return null; } else if (CollectionUtils.isNotEmpty(flags.getArtDigests())) { for (int i = 0; i < flags.getArtId().size(); i++) { if (StringUtils.isEmpty(flags.getArtDigests().get(i))) { continue; } artifacts.get(i).put("digests", Arrays.asList(StringUtils.split(flags.getArtDigests().get(i), ","))); } } if (CollectionUtils.isNotEmpty(flags.getTagKeys()) && flags.getTagKeys().size() != flags.getArtId().size()) { log.error("number of tagkey flags must be either zero or match number of artid flags"); return null; } else if (CollectionUtils.isNotEmpty(flags.getTagVals()) && flags.getTagVals().size() != flags.getArtId().size()) { log.error("number of tagval flags must be either zero or match number of artid flags"); return null; } else if (CollectionUtils.isNotEmpty(flags.getTagKeys()) && CollectionUtils.isEmpty(flags.getTagVals())) { log.error("number of tagval and tagkey flags must be the same and must match number of artid flags"); return null; } else if (CollectionUtils.isNotEmpty(flags.getTagKeys())) { for (int i = 0; i < flags.getTagKeys().size(); i++) { if (StringUtils.isEmpty(flags.getTagKeys().get(i)) || StringUtils.isEmpty(flags.getTagVals().get(i))) { continue; } List keys = Arrays.asList(StringUtils.split(flags.getTagKeys().get(i), ",")); List vals = Arrays.asList(StringUtils.split(flags.getTagVals().get(i), ",")); if (CollectionUtils.isNotEmpty(keys) && CollectionUtils.isNotEmpty(vals) && keys.size() != vals.size()) { log.error("number of keys and values per each tagval and tagkey flag must be the same"); return null; } List tags = new ArrayList<>(); for (int j = 0; j < keys.size(); j++) { tags.add(new TagRecord(keys.get(j), vals.get(j))); } artifacts.get(i).put("tags", tags); } } variables.put("artifacts", artifacts); } String query = "" + "mutation ($ReleaseInputProg: ReleaseInputProg) { \n" + "addReleaseProg(release:$ReleaseInputProg) { \n" + RELEASE_GQL_DATA + "\n" + "} \n" + "}"; Map input = new HashMap<>(); input.put("ReleaseInputProg", variables); Map body = new HashMap<>(); body.put("query", query); body.put("variables", input); Call call = rhs.addRelease(body); Map response = execute(call); return response == null ? null : OM.convertValue(response.get("addReleaseProg"), ReleaseData.class); } /** * Method that denotes we are checking artifact hash.

* Method itself does not require parameters but requires that the Flags class passed during library initialization contains these parameters.

* - apiKeyId (required) - flag for project api id.
* - apiKey (required) - flag for project api key.
* - hash (required) - flag to denote actual hash. By convention, hash must include hashing algorithm as its first part, i.e. sha256: or sha512: * @return returns ReleaseData if successful API call and null otherwise. */ public ReleaseData checkHash() { String query = "" + "query ($hash: String!) { \n" + "getReleaseByHash(hash:$hash) { \n" + RELEASE_GQL_DATA + "\n" + "} \n" + "}"; Map input = new HashMap<>(); input.put("hash", flags.getHash()); Map body = new HashMap<>(); body.put("query", query); body.put("variables", input); Call call = rhs.checkHash(body); Map response = execute(call); return response == null ? null : OM.convertValue(response.get("getReleaseByHash"), ReleaseData.class); } /** * Method that denotes that we are sending digest data from instance.

* Method itself does not require parameters but requires that the Flags class passed during library initialization contains these parameters.

* - apiKeyId (required) - flag for instance api id.
* - apiKey (required) - flag for instance api key.
* - imagesString (required) - flag which lists sha256 digests of images sent from the instances. Images must be white space separated. Note that sending full docker image URIs with digests is also accepted, i.e. it's ok to send images as relizaio/reliza-cli:latest@sha256:ebe68a0427bf88d748a4cad0a419392c75c867a216b70d4cd9ef68e8031fe7af
* - namespace (optional, if not sent "default" namespace is used) - flag to denote namespace where we are sending images. Namespaces are useful to separate different products deployed on the same instance.
* - senderId (optional) - flag to denote unique sender within a single namespace. This is useful if say there are different nodes where each streams only part of application deployment data. In this case such nodes need to use same namespace but different senders so that their data does not stomp on each other.
* @return returns status map if successful API call and null otherwise. */ public Map instData() { Map variables = new HashMap<>(); if (StringUtils.isNotEmpty(flags.getImagesString())) { variables.put("images", Arrays.asList(StringUtils.split(flags.getImagesString(), " "))); } else if (flags.getImageInputStream() != null) { try { byte[] imageBytes = IOUtils.toByteArray(flags.getImageInputStream()); variables.put("images", Arrays.asList(StringUtils.split(new String(imageBytes, StandardCharsets.UTF_8), " "))); } catch (IOException e) { log.error("IO exception", e); return null; } catch (NullPointerException e) { log.error("NullPointerException", e); return null; } } variables.put("timeSent", Instant.now().toString()); variables.put("namespace", flags.getNamespace()); variables.put("senderId", flags.getSenderId()); String query = "" + "mutation ($InstanceDataInput: InstanceDataInput) { \n" + "instData(instance:$InstanceDataInput) \n" + "}"; Map input = new HashMap<>(); input.put("InstanceDataInput", variables); Map body = new HashMap<>(); body.put("query", query); body.put("variables", input); Call call = rhs.checkHash(body); Map response = execute(call); return response == null ? null : (Map) response.get("instData"); } /** * Method that denotes that we are sending data for a pull request.

* Method itself does not require parameters but requires that the Flags class passed during library initialization contains these parameters.

* - apiKeyId (required) - flag for instance api id.
* - apiKey (required) - flag for instance api key.
* - branch (required) - flag to denote name of the base branch for the pull request.
* - targetBranch (required) - flag to denote name of the target branch for the pull request.
* - state (required) - flag to denote state of the pull request.
* - endpoint (required) - flag to denote HTML endpoint of the pull request.
* - title (required) - flag to denote title of the pull request.
* - number (required) - flag to denote number of the pull request.
* - commits (required) - flag to denote comma seprated commit shas on this pull request.
* - createdDate (required) - flag to denote datetime when the pull request was created.
* - closedDate (optional) - flag to denote datetime when the pull request was closed.
* - mergedDate (optional) - flag to denote datetime when the pull request was merged.
* - projectId (optional) - flag to denote project uuid. Required if organization-wide read-write key is used, ignored if project specific api key is used.
* @return returns true if successful API call and null otherwise. */ public Optional prData() { Map variables = new HashMap<>(); variables.put("branch", flags.getBranch()); variables.put("targetBranch", flags.getTargetBranch()); variables.put("project", flags.getProjectId()); variables.put("state", flags.getState()); variables.put("endpoint", flags.getEndPoint()); variables.put("title", flags.getTitle()); variables.put("number", flags.getNumber()); variables.put("createdDate", flags.getCreatedDate()); variables.put("closedDate", flags.getClosedDate()); variables.put("mergedDate", flags.getMergedDate()); if (StringUtils.isNotEmpty(flags.getCommits())) { String commits = new String(Base64.getDecoder().decode(flags.getCommits())); List commitList = Arrays.asList(StringUtils.split(commits, System.lineSeparator())); List> commitsInBody = new ArrayList<>(); for (int i = 0; i < commitList.size(); i++) { Map singleCommit = new HashMap<>(); List commitParts = Arrays.asList(StringUtils.split(commitList.get(i), "|||")); singleCommit.put("commit", commitParts.get(0)); singleCommit.put("dateActual", commitParts.get(1)); singleCommit.put("commitMessage", commitParts.get(2)); if (commitParts.size() > 3) { singleCommit.put("commitAuthor", commitParts.get(3)); singleCommit.put("commitEmail", commitParts.get(4)); } commitsInBody.add(singleCommit); if (i == 0 && !StringUtils.isNotEmpty(flags.getCommitHash())) { Map commitMap = new HashMap<>(); commitMap.put("commit", commitParts.get(0)); commitMap.put("dateActual", commitParts.get(1)); commitMap.put("commitMessage", commitParts.get(2)); if (commitParts.size() > 3) { commitMap.put("commitAuthor", commitParts.get(3)); commitMap.put("commitEmail", commitParts.get(4)); } commitMap.put("vcsTag", flags.getVcsTag()); commitMap.put("uri", flags.getVcsUri()); commitMap.put("type", flags.getVcsType()); variables.put("sourceCodeEntry", commitMap); } } variables.put("commits", commitsInBody); } String query = "" + "mutation ($PullRequestInput: PullRequestInput) { \n" + "setPRData(pullRequest:$PullRequestInput) \n" + "}"; Map input = new HashMap<>(); input.put("PullRequestInput", variables); Map body = new HashMap<>(); body.put("query", query); body.put("variables", input); Call call = rhs.prData(body); Map response = execute(call); Optional returnVal = Optional.empty(); if(response!= null && response.get("setPRData") != null){ returnVal = Optional.of((Boolean) response.get("setPRData")); } return returnVal; } /** * Method that denotes we are requesting release data for instance from Reliza Hub.

* Method itself does not require parameters but requires that the Flags class passed during library initialization contains these parameters.

* - apiKeyId (required) - flag for instance api id.
* - apiKey (required) - flag for instance api key.
* - namespace (optional, if not sent "default" namespace is used) - flag to denote namespace for which we are requesting release data. Namespaces are useful to separate different products deployed on the same instance. * @return returns FullRelease if successful API call and null otherwise. */ public List getMyRelease() { String query = "" + "query ($instance: String, $namespace: String) { \n" + "getMyRelease(instance:$instance, namespace:$namespace) { \n" + FULL_RELEASE_GQL_DATA + "\n" + "} \n" + "}"; Map variables = new HashMap<>(); Map body = new HashMap<>(); variables.put("instance", flags.getInstance()); variables.put("namespace", flags.getNamespace()); body.put("query", query); body.put("variables", variables); Call call = rhs.addRelease(body); Map response = execute(call); return response == null ? null : Arrays.asList(OM.convertValue(response.get("getMyRelease"), FullRelease[].class)); } /** * Method that denotes we are requesting latest release data for Project or Product from Reliza Hub.

* Method itself does not require parameters but requires that the Flags class passed during library initialization contains these parameters.

* - apiKeyId (required) - flag for api id which can be either api id for this project or organization-wide read API.
* - apiKey (required) - flag for api key which can be either api key for this project or organization-wide read API.
* - projectId (required) - flag to denote UUID of specific Project or Product, UUID must be obtained from Reliza Hub.
* - product (optional) - flag to denote UUID of Product which packages Project or Product for which we inquiry about its version via --project flag, UUID must be obtained from Reliza Hub.
* - branch (optional, if not supplied settings from Reliza Hub UI are used) - flag to denote required branch of chosen Project or Product.
* - environment (optional) - flag to denote environment to which release approvals should match. Environment can be one of: DEV, BUILD, TEST, SIT, UAT, PAT, STAGING, PRODUCTION. If not supplied, latest release will be returned regardless of approvals.
* - tagKeys (optional, if provided tagval flag must also be supplied) - flag to denote tag key to use as a selector for artifact. Note that currently only single tag is supported.
* - tagVals (optional, if provided tagkey flag must also be supplied) - flag to denote tag value to use as a selector for artifact.
* - instance (optional, if supplied namespace flag is also used and env flag gets overrided by instance's environment) - flag to denote specific instance for which release should match.
* - namespace (optional) - flag to denote specific namespace within instance, if instance is supplied. * @return returns FullRelease if successful API call and null otherwise. */ public FullRelease getLatestRelease() { Map variables = new HashMap<>(); variables.put("project", flags.getProjectId()); variables.put("environment", flags.getEnvironment()); variables.put("product", flags.getProduct()); if (CollectionUtils.isNotEmpty(flags.getTagKeys()) && CollectionUtils.isNotEmpty(flags.getTagVals())) { variables.put("tags", flags.getTagKeys().get(0) + " " + flags.getTagVals().get(0)); } variables.put("branch", flags.getBranch()); variables.put("instance", flags.getInstance()); variables.put("namespace", flags.getNamespace()); String query = "" + "query ($GetLatestReleaseInput: GetLatestReleaseInput) { \n" + "getLatestRelease(release:$GetLatestReleaseInput) { \n" + FULL_RELEASE_GQL_DATA + "\n" + "} \n" + "}"; Map input = new HashMap<>(); input.put("GetLatestReleaseInput", variables); Map body = new HashMap<>(); body.put("query", query); body.put("variables", input); Call call = rhs.addRelease(body); Map response = execute(call); return response == null ? null : OM.convertValue(response.get("getLatestRelease"), FullRelease.class); } /** * Method that denotes that we are approving release programmatically for the specific type. Must grant permission for approvalType beforehand.

* Method itself does not require parameters but requires that the Flags class passed during library initialization contains these parameters.

* - apiKeyId (required) - flag for api id.
* - apiKey (required) - flag for api key.
* - releaseId (either this flag or project id and release version is required) - flag to specify release uuid, which can be obtained from the release view or programmatically.
* - releaseVersion (either this flag and project or releaseid must be provided) - flag to specify release string version with the project flag above.
* - projectId (either this flag and release version or releaseid must be provided) - flag to specify project uuid, which can be obtained from the project settings on Reliza Hub UI.
* - instance (either instance and project or releaseid or releaseversion and project must be set) - UUID or URI of instance for which release should be approved.
* - namespace (optional, only considered if instance is specified") - Namespace of the instance for which release should be approved.
* - approvalType (required) - approval type as per approval matrix on the Organization Settings page in Reliza Hub.
* - disapprove (optional) - flag to indicate disapproval event instead of approval. * @return returns ReleaseData if successful API call and null otherwise. */ public ReleaseData approveRelease() { Map variables = new HashMap<>(); Map approvalMap = new HashMap<>(); approvalMap.put(flags.getApprovalType(), !flags.getDisapprove()); variables.put("approvals", approvalMap); variables.put("uuid", flags.getReleaseId()); variables.put("version", flags.getVersion()); variables.put("project", flags.getProjectId()); variables.put("instance", flags.getInstance()); variables.put("namespace", flags.getNamespace()); String query = "" + "mutation ($ApproveReleaseInput: ApproveReleaseInput) { \n" + "approveReleaseProg(release:$ApproveReleaseInput) { \n" + RELEASE_GQL_DATA + "\n" + "} \n" + "}"; Map input = new HashMap<>(); input.put("ApproveReleaseInput", variables); Map body = new HashMap<>(); body.put("query", query); body.put("variables", input); Call call = rhs.addRelease(body); Map response = execute(call); return response == null ? null : OM.convertValue(response.get("approveReleaseProg"), ReleaseData.class); } /** * Method for executing call and logging results * @param call - The call which we send to the API. * @return If call has no errors return API response and return null otherwise. */ private static Map execute(Call call) { try { Response resp = call.execute(); if (resp.body().getErrors() == null) { log.debug(resp.body().getData().toString()); return resp.body().getData(); } else { log.error(resp.body().getErrors().toString()); return null; } } catch (IOException e) { log.error("IO exception", e); return null; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy