2024-08-12 14:53:19 +12:00
< script >
2024-08-26 17:00:21 +12:00
// get the data from the JSON Webservices feed in +page.js
export let data;
const { webservices } = data;
2024-09-04 15:53:51 +12:00
import { references } from '$lib/references.js';
import { colours } from '$lib/colours.js';
2024-09-19 16:23:30 +12:00
// the filters
2024-09-18 15:53:10 +12:00
import { setFilter , getFilter } from '$lib/components/filter.js';
import Filterable from '$lib/components/Filterable.svelte';
2024-09-19 16:23:30 +12:00
// collapsible sections
2024-09-23 09:34:14 +12:00
import CollapsibleSection from '$lib/components/CollapsibleSection.svelte';
import Tile from '$lib/components/Tile.svelte';
2024-09-04 15:53:51 +12:00
2024-09-18 15:53:10 +12:00
//console.log('colours: ', colours);
2024-09-04 15:53:51 +12:00
2024-09-18 15:53:10 +12:00
//
2024-09-04 15:53:51 +12:00
// digest and return useful info from the Webservices JSON feed
function processData(webservices) {
2024-09-24 17:09:24 +12:00
let stats = { 'instances' : 0 , 'technologies' : 0 , 'licenses' : 0 , 'weak' : 0 , 'copyleft' : 0 } ;
let all = []; // all services
let current = []; // those with active instances
let future = []; // those that are planned
2024-09-04 15:53:51 +12:00
// properties of technologies
let category_list = [];
let analogue_list = [];
let license_list = [];
// properties of instances
let status_list = [];
let affiliate_list = [];
let host_list = [];
// actual objects
let hosts = [];
let affiliates = [];
2024-09-24 17:09:24 +12:00
2024-09-04 15:53:51 +12:00
//
// just a trivial reassignment
//let hosts = webservices.hosts;
//
// pull out relevant info in useful chunks.
for (let key in webservices.technologies) {
let tech = webservices.technologies[key];
2024-09-24 17:09:24 +12:00
tech['name'] = key;
console.log('tech '+key+' ('+tech.categories.length+'):', tech.categories);
2024-09-04 15:53:51 +12:00
if (tech.hasOwnProperty('categories') && tech.categories.constructor === Array) {
2024-09-24 17:09:24 +12:00
//console.log('tech '+key+' ('+tech.categories.length+'):', tech.categories);
2024-09-04 15:53:51 +12:00
tech.categories.forEach(function (category, index) {
if (category_list.hasOwnProperty(category)) category_list[category]++;
else category_list[category] = 1;
});
}
if (tech.hasOwnProperty('analogues') && tech.analogues.constructor === Array) {
tech.analogues.forEach(function (analogue, index) {
if (analogue_list.hasOwnProperty(analogue)) analogue_list[analogue]++;
else analogue_list[analogue] = 1;
});
}
if (tech.hasOwnProperty('license')) {
let license = tech.license;
if (license_list.hasOwnProperty(license)) license_list[license]++;
else license_list[license] = 1;
}
if (hasInstances(tech)) {
current.push(tech);
//console.log(tech.name + ': ' + tech.instances.length + ' instances...');
tech.instances.forEach(function (instance, i) {
2024-09-24 17:09:24 +12:00
stats.instances++;
2024-09-04 15:53:51 +12:00
if (instance.hasOwnProperty('status')) {
let tag = instance.status;
if (status_list.hasOwnProperty(tag)) status_list[tag]++;
else status_list[tag] = 1;
}
if (instance.hasOwnProperty('affiliation')) {
let tag = instance.affiliation;
2024-09-05 11:07:29 +12:00
2024-09-04 15:53:51 +12:00
if (affiliate_list.hasOwnProperty(tag)) affiliate_list[tag]++;
else affiliate_list[tag] = 1;
}
if (instance.hasOwnProperty('host')) {
let tag = instance.host;
if (host_list.hasOwnProperty(tag)) host_list[tag]++;
else host_list[tag] = 1;
}
});
} else {
future[key] = tech;
}
2024-09-24 17:09:24 +12:00
all[key] = tech;
2024-09-04 15:53:51 +12:00
}
for (let key in webservices.hosts) {
//console.log('key: ', key);
let host = webservices.hosts[key];
host['name'] = key;
//console.log('host: ', host);
if (
host.hasOwnProperty('domain') & &
!(host.hasOwnProperty('status') & & host.status == 'retired')
) {
hosts[key] = host;
}
}
for (let key in webservices.affiliates) {
let affiliate = webservices.affiliates[key];
affiliates[key] = affiliate;
}
2024-09-24 17:09:24 +12:00
// assign stats based on gathered data...
stats.current = current.length;
stats.future = future.length;
console.log('license_list: ', license_list);
stats.licenses = license_list.length;
2024-09-04 15:53:51 +12:00
return {
2024-09-24 17:09:24 +12:00
stats: stats,
all_services: all,
2024-09-04 15:53:51 +12:00
active_services: current,
candidate_services: future,
tech_lists: {
category_list: category_list,
analogue_list: analogue_list,
license_list: license_list
},
instance_lists: {
status_list: status_list,
affiliate_list: affiliate_list,
host_list: host_list
},
hosts: hosts,
affiliates: affiliates
};
}
2024-08-26 17:00:21 +12:00
2024-09-24 17:09:24 +12:00
// return an object with ob's keys ordered alphabetically, with an id from an object
2024-09-18 15:53:10 +12:00
function getSortedFilter(ob) {
2024-08-26 17:00:21 +12:00
let keys = [];
2024-09-18 15:53:10 +12:00
let i = 0;
let key_array = [];
// first create a flat array.
2024-08-26 17:00:21 +12:00
for (let key in ob) {
keys.push(key);
}
2024-09-18 15:53:10 +12:00
// sort the array alphabetically
keys.sort();
// then create a dictionary of them
keys.forEach(function(name) {
key_array.push({ "id" : i ++ , "name" : name , "active" : true } );
});
return key_array;
2024-08-26 17:00:21 +12:00
}
2024-09-24 17:09:24 +12:00
// return true if a tech object includes valid instances
function hasInstances(tech) {
if (
tech.hasOwnProperty('instances') & &
tech.instances.constructor === Array & &
tech.instances.length
) return true;
return false;
}
2024-08-26 17:00:21 +12:00
// get intersection: ref https://bobbyhadz.com/blog/javascript-get-intersection-of-two-arrays
function getIntersection(a, b) {
const set1 = new Set(a);
const set2 = new Set(b);
const intersection = [...set1].filter((element) => set2.has(element));
return intersection;
}
// assign colours from a set of differentiated colours to a list of tags...
// this one is for 'hosts'
2024-09-18 15:53:10 +12:00
//
// host_list is in the form of
2024-08-26 17:00:21 +12:00
function hostColours(host_list, colours, hosts) {
let host_array = {} ;
let i = 0;
2024-09-18 15:53:10 +12:00
console.log('hosts:', hosts);
console.log('host_list:', host_list);
2024-08-26 17:00:21 +12:00
2024-09-18 15:53:10 +12:00
for (let index in host_list) {
console.log(host_list[index]);
let hostname = host_list[index].name;
if (hosts[hostname].hasOwnProperty('domain') && hosts[hostname].hasOwnProperty('affiliation')) {
host_array[hostname] = {
2024-08-26 17:00:21 +12:00
colour: colours[i++],
2024-09-18 15:53:10 +12:00
domain: hosts[hostname].domain,
affiliation: hosts[hostname].affiliation
2024-08-26 17:00:21 +12:00
};
}
2024-09-18 15:53:10 +12:00
}
2024-08-26 17:00:21 +12:00
return host_array;
}
// assign colours from a set of differentiated colours to a list of tags...
// this one is for 'affiliates'
function affiliateColours(affiliate_list, colours, affiliates) {
let affiliate_result = {} ;
let i = 0;
//console.log('affiliates:', affiliates);
console.log('affiliate_list:', affiliate_list);
2024-09-18 15:53:10 +12:00
for (let index in affiliate_list) {
2024-08-26 17:00:21 +12:00
//console.log('affiliate:', affiliate);
2024-09-18 15:53:10 +12:00
let affiliate = affiliate_list[index].name;
2024-08-26 17:00:21 +12:00
if (
affiliates[affiliate].hasOwnProperty('name') & &
affiliates[affiliate].hasOwnProperty('website')
) {
affiliate_result[affiliate] = {
colour: colours[i++],
name: affiliates[affiliate].name,
website: affiliates[affiliate].website
};
}
}
return affiliate_result;
}
// sort technologies alphabetically by name
function sortTechnologies(technologies) {
return technologies.sort((a, b) => a.name.localeCompare(b.name));
}
// generic function to find the intersection of two arrays
// approach ref: https://bobbyhadz.com/blog/javascript-get-intersection-of-two-arrays
function inCommon(a, b) {
const set1 = new Set(a);
const set2 = new Set(b);
const intersection = [...set1].filter((element) => set2.has(element));
2024-09-24 17:09:24 +12:00
//console.log('intersection = ', intersection);
2024-08-26 17:00:21 +12:00
return intersection;
}
2024-09-19 16:23:30 +12:00
// converts a filter with id, name, and active fields into an array of names
function flattenFilter(filter) {
let flat = [];
if (filter.constructor === Array) {
filter.forEach(function(e) { if ( e . active ) flat . push ( e . name ) } );
}
return flat;
}
2024-08-26 17:00:21 +12:00
// filter technologies based on a list of Categories
function filterTechnologiesByCategoryList(technologies, list) {
2024-09-24 17:09:24 +12:00
//console.log('looking for tech in categories: ', list);
2024-08-26 17:00:21 +12:00
const included = [];
2024-09-19 16:23:30 +12:00
technologies.forEach(function (tech) {
2024-08-26 17:00:21 +12:00
if (hasInstances(tech)) {
const intersection = inCommon(tech.categories, list);
2024-09-24 17:09:24 +12:00
console.log('for tech: ', tech.name);
console.log('categories: ', tech.categories);
console.log('list: ', list);
2024-09-19 16:23:30 +12:00
console.log('intersection: ', intersection);
2024-08-26 17:00:21 +12:00
if (intersection && intersection.constructor === Array) {
2024-09-19 16:23:30 +12:00
//console.log('found intersection!', intersection.constructor);
console.log('intersection length: ', intersection.length);
2024-08-26 17:00:21 +12:00
if (intersection.length > 0) {
2024-09-18 15:53:10 +12:00
//console.log('including tech: ', tech);
2024-08-26 17:00:21 +12:00
included.push(tech);
}
} else {
2024-09-19 16:23:30 +12:00
console.log('intersection not array');
2024-08-26 17:00:21 +12:00
}
}
});
console.log('found ' + included.length + ' technologies.');
return included;
}
const results = processData(webservices);
//const technologies = webservices.technologies;
//console.log(technologies);
2024-09-18 15:53:10 +12:00
// set up filters for each type
setFilter('categories');
setFilter('analogues');
setFilter('licenses');
setFilter('statuses');
setFilter('affiliates');
setFilter('hosts');
// define the handle for each writable
const categoryFilter = getFilter('categories');
const analogueFilter = getFilter('analogues');
const licenseFilter = getFilter('licenses');
const statusFilter = getFilter('statuses');
const affiliateFilter = getFilter('affiliates');
const hostFilter = getFilter('hosts');
// populate the writable with actual data
$categoryFilter = getSortedFilter(results.tech_lists.category_list);
$analogueFilter = getSortedFilter(results.tech_lists.analogue_list);
$licenseFilter = getSortedFilter(results.tech_lists.license_list);
$statusFilter = getSortedFilter(results.instance_lists.status_list);
$affiliateFilter = getSortedFilter(results.instance_lists.affiliate_list);
$hostFilter = getSortedFilter(results.instance_lists.host_list);
2024-08-26 17:00:21 +12:00
const hosts = results.hosts;
2024-09-18 15:53:10 +12:00
console.log('host_data: ', results.hosts);
2024-08-26 17:00:21 +12:00
const affiliates = results.affiliates;
2024-09-18 15:53:10 +12:00
console.log('affiliate_data: ', results.affiliates);
const host_colours = hostColours($hostFilter, colours, hosts);
console.log('host_colours:', host_colours);
2024-08-30 15:23:00 +12:00
colours.sort();
2024-09-18 15:53:10 +12:00
const affiliate_colours = affiliateColours($affiliateFilter, colours, affiliates);
console.log('affiliate_colours:',affiliate_colours);
2024-08-26 17:00:21 +12:00
2024-09-19 16:23:30 +12:00
let filteredTechnologies;
let technologies;
2024-08-26 17:00:21 +12:00
2024-09-19 16:23:30 +12:00
// reactive stuff...
$: {
2024-09-24 17:09:24 +12:00
//console.log('about to filterTechnologiesByCategoryList');
2024-09-19 16:23:30 +12:00
filteredTechnologies = filterTechnologiesByCategoryList(results.active_services, flattenFilter($categoryFilter));
2024-09-24 17:09:24 +12:00
//filteredTechnologies = filterTechnologiesByCategoryList(results.all_services, flattenFilter($categoryFilter));
2024-09-19 16:23:30 +12:00
const technologies = sortTechnologies(filteredTechnologies);
}
2024-09-18 15:53:10 +12:00
2024-09-19 16:23:30 +12:00
/*function flipOn(list, id) {
2024-09-18 15:53:10 +12:00
console.log('list: ' + list + ', id: ' + id);
return true;
2024-09-19 16:23:30 +12:00
}*/
2024-09-18 15:53:10 +12:00
2024-08-12 14:53:19 +12:00
< / script >
< div class = "webservices" >
2024-09-24 17:09:24 +12:00
< div class = "header" >
< div class = "introduction" >
< h1 > Web Services< / h1 >
< p > This sites exists to provide an 'always-up-to-date', in-depth description of the < a
href="https://tech.oeru.org/foss-libresoftware-its-about-clarity-and-values"
title="What do I mean by 'libre' software?">libre software< / a > web services < a href = "https://davelane.nz" title = "Who is this Dave Lane character?" > I< / a > have set
up and maintain.
< / p >
2024-09-04 15:53:51 +12:00
2024-09-24 17:09:24 +12:00
< / div >
< div class = "summary" >
< h2 > Statistics< / h2 >
< ul >
< li > Number of individual services: { results . stats . instances } </ li >
< li > Number of individual technologies: { results . stats . current } </ li >
< li > Number of new technologies to be added: { results . stats . future } </ li >
< li > Number of libre licenses used by these technologies: { results . stats . licenses } </ li >
< li > 'weak' corporate-exploitation-friendly open source licenses : { results . stats . weak } </ li >
< li > 'strong' user-protecting copyleft licenses: { results . stats . copyleft } </ li >
< li > Total number of services: { results . stats . instances } </ li >
< / ul >
< / div >
2024-08-26 17:00:21 +12:00
< / div >
2024-09-23 09:34:14 +12:00
< CollapsibleSection headerText = { 'Filters' } >
2024-09-19 16:23:30 +12:00
< div class = "filters" >
2024-09-24 17:09:24 +12:00
< CollapsibleSection headerText = { 'Software Catergory Filter' } >
2024-09-23 09:34:14 +12:00
< div class = "filter-group categories" >
2024-09-24 17:09:24 +12:00
< Filterable filterValues = { categoryFilter }
histogram={ results . tech_lists . category_list } context={ 'categories' } >
2024-09-19 16:23:30 +12:00
< h2 slot = "h2" > Categories< / h2 >
< / Filterable >
< / div >
< / CollapsibleSection >
2024-09-23 09:34:14 +12:00
< CollapsibleSection headerText = { 'Software Analogue Filter' } >
< div class = "filter-group analogues" >
2024-09-24 17:09:24 +12:00
< Filterable filterValues = { analogueFilter }
histogram={ results . tech_lists . analogue_list } context={ 'analogues' } >
2024-09-19 16:23:30 +12:00
< h2 slot = "h2" > Analogues< / h2 >
< / Filterable >
< / div >
< / CollapsibleSection >
2024-09-23 09:34:14 +12:00
< CollapsibleSection headerText = { 'License Filter' } >
< div class = "filter-group licenses" >
2024-09-24 17:09:24 +12:00
< Filterable filterValues = { licenseFilter }
histogram={ results . tech_lists . license_list } context={ 'licenses' } >
2024-09-19 16:23:30 +12:00
< h2 slot = "h2" > Open Source & Copyleft Licenses< / h2 >
< / Filterable >
< / div >
< / CollapsibleSection >
2024-09-23 09:34:14 +12:00
< CollapsibleSection headerText = { 'Service Status Filter' } >
< div class = "filter-group statuses" >
2024-09-24 17:09:24 +12:00
< Filterable filterValues = { statusFilter }
histogram={ results . instance_lists . status_list } context={ 'statuses' } >
2024-09-19 16:23:30 +12:00
< h2 slot = "h2" > Statuses< / h2 >
< / Filterable >
< / div >
< / CollapsibleSection >
2024-09-23 09:34:14 +12:00
< CollapsibleSection headerText = { 'Service Affiliates Filter' } >
< div class = "filter-group affilates" >
2024-09-24 17:09:24 +12:00
< Filterable filterValues = { affiliateFilter }
histogram={ results . instance_lists . affiliate_list } context={ 'affiliates' } >
2024-09-19 16:23:30 +12:00
< h2 slot = "h2" > Affiliates< / h2 >
< / Filterable >
< / div >
< / CollapsibleSection >
2024-09-23 09:34:14 +12:00
< CollapsibleSection headerText = { 'Service Host filter' } >
< div class = "filter-group hosts" >
2024-09-24 17:09:24 +12:00
< Filterable filterValues = { hostFilter }
histogram={ results . instance_lists . host_list } context={ 'hosts' } >
2024-09-19 16:23:30 +12:00
< h2 slot = "h2" > Hosts< / h2 >
< / Filterable >
< / div >
< / CollapsibleSection >
2024-08-26 17:00:21 +12:00
< / div >
2024-09-19 16:23:30 +12:00
< / CollapsibleSection >
2024-08-26 17:00:21 +12:00
< div class = "tiles" >
2024-09-19 16:23:30 +12:00
{ #each filteredTechnologies as technology }
2024-09-24 17:09:24 +12:00
< Tile technology = { technology } affiliate_colours= { affiliate_colours } />
2024-08-26 17:00:21 +12:00
{ /each }
< / div >
2024-08-12 14:53:19 +12:00
< / div >
2024-08-16 17:01:16 +12:00
< style >
2024-08-26 17:00:21 +12:00
.webservices {
display: grid;
margin: 0 auto 3em auto;
padding: 0;
2024-09-24 17:09:24 +12:00
position: relative;
}
.introduction {
width: 50%;
2024-08-26 17:00:21 +12:00
}
.summary {
2024-09-24 17:09:24 +12:00
border-radius: 15px;
/*position: absolute;
right: 20px;*/
2024-08-26 17:00:21 +12:00
background-color: #ccc;
2024-09-24 17:09:24 +12:00
/*display: block;*/
width: 50%;
border: 2px #aaa solid;
margin: 0;
padding: 0 20px;
2024-08-26 17:00:21 +12:00
}
2024-09-23 09:34:14 +12:00
/* .filters { margin - bottom : 1em ;}
2024-08-26 17:00:21 +12:00
.filters .tags {
margin: 1em 0;
display: block;
2024-09-23 09:34:14 +12:00
}*/
2024-08-26 17:00:21 +12:00
.tiles {
2024-09-24 17:09:24 +12:00
/*display: grid;
2024-08-26 17:00:21 +12:00
grid-gap: 15px;
2024-09-24 17:09:24 +12:00
grid-template-columns: repeat(auto-fit, minmax(270px, 1fr));*/
/*grid-template-columns: repeat(auto-fit);*/
2024-08-26 17:00:21 +12:00
}
2024-08-12 14:53:19 +12:00
< / style >