Skip to content

Commit

Permalink
Create instance
Browse files Browse the repository at this point in the history
  • Loading branch information
pbochynski committed Nov 14, 2024
1 parent 98ee803 commit 84bf1a3
Showing 1 changed file with 256 additions and 32 deletions.
288 changes: 256 additions & 32 deletions examples/web-component-ext/script.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,27 +4,66 @@ class MyComponent1 extends HTMLElement {
let style = styling();
shadow.appendChild(style);

// create elements programmatically
const button = document.createElement('ui5-button');
button.textContent = 'Load offerings';
button.design = 'Emphasized';
shadow.appendChild(button);
const contentDiv = document.createElement('div');
shadow.appendChild(contentDiv);
button.addEventListener('click', async () => {
const fetchFn = window.kymaFetchFn;
if (!fetchFn) {
alert('Kyma fetch function is not available');
return;
}
let secret = await getSMsecret();
let token = await getServiceManagerToken(secret);
let offerings = await getServiceOfferings(token, secret.sm_url);
contentDiv.innerHTML = '';
for (let i = 0; i < offerings.length; i++) {
contentDiv.appendChild(offeringCard(offerings[i], i, offerings.length));
}
});
contentDiv.innerHTML = 'loading...';
loadData(contentDiv);
}
}
function sortByInstanceCountAndOfferingName(a, b) {
if (a.instances.length > b.instances.length) {
return -1;
}
if (a.instances.length < b.instances.length) {
return 1;
}
if (a.metadata.displayName < b.metadata.displayName) {
return -1;
}
if (a.metadata.displayName > b.metadata.displayName) {
return 1;
}
return 0;
}
async function loadData(contentDiv) {
let secret = await getSMsecret();
let token = await getServiceManagerToken(secret);
const tasks = [
getServiceOfferings(token, secret.sm_url),
getServicePlans(token, secret.sm_url),
getServiceInstances(),
getServiceBindings(),
getNamespaces(),
];
let results = await Promise.allSettled(tasks);
let offerings = results[0].value;
let plans = results[1].value;
let instances = results[2].value;
let bindings = results[3].value;
let namespaces = results[4].value;
console.log(offerings, plans, instances, bindings, namespaces);
for (let offering of offerings) {
offering.namespaces = namespaces;
offering.instances = instances.filter(
instance => instance.spec.serviceOfferingName === offering.name,
);
offering.plans = plans.filter(
plan => plan.service_offering_id === offering.id,
);
for (let instance of offering.instances) {
instance.bindings = bindings.filter(
binding =>
binding.spec.serviceInstanceName === instance.metadata.name &&
binding.metadata.namespace === instance.metadata.namespace,
);
}
}
offerings = offerings.sort(sortByInstanceCountAndOfferingName);

contentDiv.innerHTML = '';
console.log(offerings);
for (let offering of offerings) {
contentDiv.appendChild(offeringCard(offering));
}
}

Expand All @@ -34,6 +73,11 @@ function styling() {
.content-group {
padding: 1rem;
}
.popover-content {
display: flex;
flex-direction: column;
gap: 1rem;
}
`;
return style;
}
Expand All @@ -58,6 +102,136 @@ async function getSMsecret() {
return secret;
}

async function createServiceInstance(name, namespace, offering, plan) {
const serviceInstance = {
apiVersion: 'services.cloud.sap.com/v1',
kind: 'ServiceInstance',
metadata: {
name: name,
namespace: namespace,
},
spec: {
serviceOfferingName: offering,
servicePlanName: plan,
},
};
let relativeUrl = `/apis/services.cloud.sap.com/v1/namespaces/${namespace}/serviceinstances`;
try {
let resp = await window.kymaFetchFn({
relativeUrl,
init: {
method: 'POST',
body: JSON.stringify(serviceInstance),
headers: {
'Content-Type': 'application/json',
},
},
});
console.log(resp);
return { status: 'ok', message: 'Service instance created' };
} catch (e) {
return {
status: 'error',
message: `Failed to create service instance: ${e}`,
};
}
}

async function getServiceInstances() {
let relativeUrl = '/apis/services.cloud.sap.com/v1/serviceinstances';
let resp = await window.kymaFetchFn({ relativeUrl });
let data = await resp.json();
return data.items;
}
async function getServiceBindings() {
let relativeUrl = '/apis/services.cloud.sap.com/v1/servicebindings';
let resp = await window.kymaFetchFn({ relativeUrl });
let data = await resp.json();
return data.items;
}
async function getNamespaces() {
let relativeUrl = '/api/v1/namespaces';
let resp = await window.kymaFetchFn({ relativeUrl });
let data = await resp.json();
return data.items;
}

function label(text, forId, required) {
const label = document.createElement('ui5-label');
label.setAttribute('for', forId);
if (required) {
label.setAttribute('required', '');
}
label.innerHTML = text;
return label;
}
function dropdown(id, items) {
const dropdown = document.createElement('ui5-select');
dropdown.id = id;
for (let item of items) {
const option = document.createElement('ui5-option');
option.value = item;
option.innerHTML = item;
dropdown.appendChild(option);
}
return dropdown;
}

function createPopover(namespaces, plans, onOk) {
const popover = document.createElement('ui5-popover');
popover.setAttribute('header-text', 'Create Instance');
popover.setAttribute('placement', 'Bottom');
const content = document.createElement('div');
content.setAttribute('class', 'popover-content');
content.appendChild(label('Name', 'instanceName', true));
const instanceName = document.createElement('ui5-input');
instanceName.id = 'instanceName';
content.appendChild(instanceName);

content.appendChild(label('Namespace', 'namespaceDropdown', true));
const namespaceDropdown = dropdown(
'namespaceDropdown',
namespaces.map(ns => ns.metadata.name),
);
content.appendChild(namespaceDropdown);

content.appendChild(label('Plan', 'planDropdown', true));
const planDropdown = dropdown(
'planDropdown',
plans.map(plan => plan.name),
);
content.appendChild(planDropdown);

popover.appendChild(content);
const footer = document.createElement('div');
footer.setAttribute('slot', 'footer');
footer.setAttribute('class', 'popover-footer');
const closePopoverButton = document.createElement('ui5-button');
closePopoverButton.id = 'closePopoverButton';
closePopoverButton.setAttribute('design', 'Emphasized');
closePopoverButton.innerHTML = 'Close';
closePopoverButton.addEventListener('click', () => {
popover.open = false;
});
footer.appendChild(closePopoverButton);
const createInstanceBtn = document.createElement('ui5-button');
createInstanceBtn.id = 'createInstanceBtn';
createInstanceBtn.setAttribute('design', 'Emphasized');
createInstanceBtn.innerHTML = 'Create Instance';
createInstanceBtn.addEventListener('click', () => {
const namespace = namespaceDropdown.value;
const plan = planDropdown.value;

onOk(instanceName.value, namespace, plan);
popover.open = false;
});

footer.appendChild(createInstanceBtn);
popover.appendChild(footer);

return popover;
}

async function getServiceManagerToken(secret) {
const client_id = secret.clientid;
const client_secret = secret.clientsecret;
Expand Down Expand Up @@ -86,31 +260,40 @@ async function getServiceOfferings(token, url) {
let data = await resp.json();
return data.items;
}

function getProxyAddress() {
// dev busola
if (window.location.hostname.startsWith('localhost')) {
return 'http://localhost:3001/proxy';
// on cluster
} else {
return '/proxy';
}
async function getServicePlans(token, url) {
let resp = await proxyFetch(url + `/v1/service_plans`, {
method: 'GET',
headers: {
authorization: `Bearer ${token}`,
},
});
let data = await resp.json();
return data.items;
}

function proxyFetch(url, options = {}) {
let proxyUrl = getProxyAddress() + `?url=${url}`;
let baseUrl = '/proxy';
if (window.location.hostname.startsWith('localhost')) {
baseUrl = 'http://localhost:3001/proxy';
}
let proxyUrl = baseUrl + `?url=${url}`;
return fetch(proxyUrl, options);
}

function offeringCard(offering, index, total) {
function offeringCard(offering) {
const greyImg =
'';
const msg = toast('ok', 'ok');
let card = document.createElement('ui5-card');
card.appendChild(msg);
let header = document.createElement('ui5-card-header');
// header.slot = 'header';
header.setAttribute('title-text', offering.metadata.displayName);
header.setAttribute('subtitle-text', offering.catalog_name);
header.setAttribute('status', `${index + 1} of ${total}`);
const status = `${offering.instances.length} instance${
offering.instances.length === 1 ? '' : 's'
}`;
header.setAttribute('status', status);
const avatar = document.createElement('ui5-avatar');
avatar.slot = 'avatar';
const img = document.createElement('img');
Expand All @@ -120,8 +303,49 @@ function offeringCard(offering, index, total) {

let cardContent = document.createElement('div');
cardContent.setAttribute('class', 'content-group');
cardContent.innerHTML = `${offering.description}`;
cardContent.innerHTML = `${offering.description}<br/>`;

const { btn, popover } = createInstanceButton(
offering,
async (name, namespace, plan) => {
console.log(name, namespace, plan);
const resp = await createServiceInstance(
name,
namespace,
offering.name,
plan,
);
msg.innerText = `${resp.status}: ${resp.message}`;
msg.open = true;
},
);
cardContent.appendChild(btn);
cardContent.appendChild(popover);

card.appendChild(header);
card.appendChild(cardContent);
return card;
}
function toast(status, message) {
const toast = document.createElement('ui5-toast');
toast.setAttribute('duration', '5000');
toast.setAttribute('placement', 'TopCenter');
toast.innerText = `${status}: ${message}`;
return toast;
}

function createInstanceButton(offering, onOk) {
const btn = document.createElement('ui5-button');
btn.setAttribute('icon', 'add');
btn.innerHTML = 'Create Instance';
btn.setAttribute('design', 'Emphasized');
//set id to offering name
btn.id = 'btn_' + offering.name;
const popover = createPopover(offering.namespaces, offering.plans, onOk);

btn.onclick = () => {
popover.setAttribute('opener', btn.id);
popover.open = true;
};
return { btn, popover };
}

0 comments on commit 84bf1a3

Please sign in to comment.