Files
snappymail/plugins/nextcloud/js/webdav.js
2024-10-07 00:03:35 +02:00

503 lines
14 KiB
JavaScript

(rl => {
const
nsDAV = 'DAV:',
nsNC = 'http://nextcloud.org/ns',
nsOC = 'http://owncloud.org/ns',
nsOCS = 'http://open-collaboration-services.org/ns',
nsCalDAV = 'urn:ietf:params:xml:ns:caldav',
OC = () => parent.OC,
// Nextcloud 19 deprecated generateUrl, but screw `import { generateUrl } from "@nextcloud/router"`
shareUrl = () => OC().webroot + '/ocs/v2.php/apps/files_sharing/api/v1/shares',
generateUrl = path => OC().webroot + (OC().config.modRewriteWorking ? '' : '/index.php') + path,
generateRemoteUrl = path => location.protocol + '//' + location.host + generateUrl(path),
// shareTypes = {0 = user, 1 = group, 3 = public link}
propfindFiles = `<?xml version="1.0"?>
<propfind xmlns="DAV:" xmlns:oc="${nsOC}" xmlns:ocs="${nsOCS}" xmlns:nc="${nsNC}">
<prop>
<oc:fileid/>
<oc:size/>
<resourcetype/>
<getcontentlength/>
<ocs:share-permissions/>
<oc:share-types/>
</prop>
</propfind>`,
propfindCal = `<?xml version="1.0"?>
<d:propfind xmlns:d="DAV:" xmlns:x1="http://apple.com/ns/ical/">
<d:prop>
<d:resourcetype/>
<d:current-user-privilege-set/>
<d:displayname/>
<x1:calendar-color/>
</d:prop>
</d:propfind>`,
xmlParser = new DOMParser(),
pathRegex = /.*\/remote.php\/dav\/[^/]+\/[^/]+/g,
getElementsByTagName = (parent, namespace, localName) => parent.getElementsByTagNameNS(namespace, localName),
getDavElementsByTagName = (parent, localName) => getElementsByTagName(parent, nsDAV, localName),
getDavElementByTagName = (parent, localName) => getDavElementsByTagName(parent, localName)?.item(0),
getElementByTagName = (parent, localName) => +parent.getElementsByTagName(localName)?.item(0),
ncFetch = (path, options) => {
if (!OC().requestToken) {
return Promise.reject(new Error('OC.requestToken missing'));
}
options = Object.assign({
mode: 'same-origin',
cache: 'no-cache',
redirect: 'error',
credentials: 'same-origin',
headers: {}
}, options);
options.headers.requesttoken = OC().requestToken;
return fetch(path, options);
},
davFetch = (mode, path, options) => {
let cfg = rl.settings.get('Nextcloud');
// cfg.UID = document.head.dataset.user
return ncFetch(cfg.WebDAV + '/' + mode + '/' + cfg.UID + path, options);
},
davFetchFiles = (path, options) => davFetch('files', path, options),
createDirectory = path => davFetchFiles(path, { method: 'MKCOL' }),
fetchFiles = path => {
if (!OC().requestToken) {
return Promise.reject(new Error('OC.requestToken missing'));
}
return davFetchFiles(path, {
method: 'PROPFIND',
headers: {
'Content-Type': 'application/xml; charset=utf-8'
},
body: propfindFiles
})
.then(response => (response.status < 400) ? response.text() : Promise.reject(new Error({ response })))
.then(text => {
const
elemList = [],
responseList = getDavElementsByTagName(
xmlParser.parseFromString(text, 'application/xml').documentElement,
'response'
);
path = path.replace(/\/$/, '');
for (let i = 0; i < responseList.length; ++i) {
const
e = responseList.item(i),
elem = {
name: decodeURIComponent(getDavElementByTagName(e, 'href').textContent
.replace(pathRegex, '').replace(/\/$/, '')),
isFile: false
};
if (getDavElementsByTagName(getDavElementByTagName(e, 'resourcetype'), 'collection').length) {
// skip current directory
if (elem.name === path) {
continue;
}
} else {
elem.isFile = true;
elem.id = e.getElementsByTagNameNS(nsOC, 'fileid')?.item(0)?.textContent;
elem.size = getDavElementByTagName(e, 'getcontentlength')?.textContent
|| getElementByTagName(e, 'oc:size')?.textContent;
elem.shared = [...e.getElementsByTagNameNS(nsOC, 'share-type')].some(node => '3' == node.textContent);
}
elemList.push(elem);
}
// https://github.com/the-djmaze/snappymail/issues/1177
// elemList.sort((a, b) => a.isFile != b.isFile ? (a.isFile ? 1 : -1) : a.name.localeCompare(b.name));
return Promise.resolve(elemList);
});
},
buildTree = (view, parent, items, path) => {
if (items.length) {
try {
// https://github.com/the-djmaze/snappymail/issues/1109
let collator = new Intl.Collator(undefined, {numeric: true, sensitivity: 'base'});
items.sort((a, b) => collator.compare(a.name, b.name));
} catch (e) {
console.error(e);
}
items.forEach(item => {
if (!item.isFile) {
let li = document.createElement('li'),
details = document.createElement('details'),
summary = document.createElement('summary'),
ul = document.createElement('ul');
details.addEventListener('toggle', () => {
ul.children.length
|| fetchFiles(item.name).then(items => buildTree(view, ul, items, item.name));
});
summary.textContent = item.name.replace(/^.*\/([^/]+)$/, '$1');
summary.dataset.icon = '📁';
if (!view.files()) {
let btn = document.createElement('button');
btn.name = 'select';
btn.textContent = 'select';
btn.className = 'button-vue';
summary.append(btn);
summary.item_name = item.name;
}
details.append(summary);
details.append(ul);
// a.append('- ' + item.name.replace(/^\/+/, ''));
li.append(details);
parent.append(li);
}
});
if (view.files()) {
items.forEach(item => {
if (item.isFile) {
let li = document.createElement('li'),
cb = document.createElement('input');
li.item = item;
li.textContent = item.name.replace(/^.*\/([^/]+)$/, '$1');
li.dataset.icon = '🗎';
cb.type = 'checkbox';
li.append(cb);
parent.append(li);
}
});
}
}
if (!view.files()) {
let li = document.createElement('li'),
input = document.createElement('input'),
btn = document.createElement('button');
btn.name = 'create';
btn.textContent = 'create & select';
btn.className = 'button-vue';
btn.input = input;
li.item_path = path;
li.append(input);
li.append(btn);
parent.append(li);
}
};
class NextcloudFilesPopupView extends rl.pluginPopupView {
constructor() {
super('NextcloudFiles');
this.addObservables({
files: false
});
}
attach() {
this.select = [];
this.tree.querySelectorAll('input').forEach(input =>
input.checked && this.select.push(input.parentNode.item)
);
this.close();
}
shareInternal() {
this.select = [];
this.tree.querySelectorAll('input').forEach(input =>
input.checked && this.select.push({url:generateRemoteUrl(`/f/${input.parentNode.item.id}`)})
);
this.close();
}
sharePublic() {
const inputs = [...this.tree.querySelectorAll('input')],
loop = () => {
if (!inputs.length) {
this.close();
return;
}
const input = inputs.pop();
if (!input.checked) {
loop();
} else {
const item = input.parentNode.item;
if (item.shared) {
ncFetch(
shareUrl() + `?format=json&path=${encodeURIComponent(item.name)}&reshares=true`
)
.then(response => (response.status < 400) ? response.json() : Promise.reject(new Error({ response })))
.then(json => {
this.select.push({url:json.ocs.data[0].url});
loop();
// json.data[0].password
});
} else {
ncFetch(
shareUrl(),
{
method:'POST',
headers: {
Accept: 'application/json',
'Content-Type': 'application/json'
},
body: JSON.stringify({
path:item.name,
shareType:3,
attributes:"[]"
})
}
)
.then(response => (response.status < 400) ? response.json() : Promise.reject(new Error({ response })))
.then(json => {
// PUT /ocs/v2.php/apps/files_sharing/api/v1/shares/2 {"password":"ABC09"}
this.select.push({url:json.ocs.data.url});
loop();
});
}
}
};
this.select = [];
loop();
}
onBuild(dom) {
this.tree = dom.querySelector('#sm-nc-files-tree');
this.tree.addEventListener('click', event => {
let el = event.target;
if (el.matches('button')) {
let parent = el.parentNode;
if ('select' == el.name) {
this.select = parent.item_name;
this.close();
} else if ('create' == el.name) {
let name = el.input.value.replace(/[|\\?*<":>+[]\/&\s]/g, '');
if (name.length) {
name = parent.item_path + '/' + name;
createDirectory(name).then(response => {
if (response.status == 201) {
this.select = name;
this.close();
}
});
}
}
}
});
}
// Happens after showModal()
beforeShow(files, fResolve) {
this.select = '';
this.files(!!files);
this.fResolve = fResolve;
this.tree.innerHTML = '';
fetchFiles('/').then(items => {
buildTree(this, this.tree, items, '/');
}).catch(err => console.error(err))
}
onHide() {
this.fResolve(this.select);
}
/*
beforeShow() {} // Happens before showModal()
onShow() {} // Happens after showModal()
afterShow() {} // Happens after showModal() animation transitionend
onHide() {} // Happens before animation transitionend
afterHide() {} // Happens after animation transitionend
close() {}
*/
}
class NextcloudCalendarsPopupView extends rl.pluginPopupView {
constructor() {
super('NextcloudCalendars');
}
onBuild(dom) {
this.tree = dom.querySelector('#sm-nc-calendars');
this.tree.addEventListener('click', event => {
let el = event.target;
if (el.matches('button')) {
this.select = el.href;
this.close();
}
});
}
createCalendarListItem(calendarData, treeElement) {
const {
displayName,
href,
calendarColor
} = calendarData;
const li = document.createElement('li');
li.style.display = 'flex';
const span = document.createElement('span');
span.setAttribute('role', 'img');
span.className = 'material-design-icon checkbox-blank-circle-icon';
span.style.fill = calendarColor;
span.style.width = '20px';
span.style.height = '20px';
const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg');
svg.setAttribute('width', '20');
svg.setAttribute('height', '20');
svg.setAttribute('viewBox', '0 0 24 24');
const path = document.createElementNS('http://www.w3.org/2000/svg', 'path');
path.setAttribute('d', 'M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10,10 0 0,0 12,2Z');
svg.appendChild(path);
span.appendChild(svg);
const button = document.createElement('button');
button.className = 'button-vue';
button.style.backgroundColor = 'transparent';
button.style.border = '0';
button.style.fontSize = 'large';
button.style.padding = '0';
button.style.cursor = 'pointer';
button.style.marginLeft = '5px';
button.href = href.replace(pathRegex, '').replace(/\/$/, '');
button.textContent = displayName;
button.style.color = '#1968DF';
li.appendChild(span);
li.appendChild(button);
treeElement.appendChild(li);
}
// Happens after showModal()
beforeShow(fResolve) {
this.select = '';
this.fResolve = fResolve;
this.tree.innerHTML = '';
davFetch('calendars', '/', {
method: 'PROPFIND',
headers: {
'Content-Type': 'application/xml; charset=utf-8'
},
body: propfindCal
})
.then(response => (response.status < 400) ? response.text() : Promise.reject(new Error({ response })))
.then(text => {
// Parse the XML text
const xmlDoc = xmlParser.parseFromString(text, 'application/xml').documentElement;
const responseList = getElementsInNamespaces(xmlDoc, 'response');
for (let i = 0; i < responseList.length; ++i) {
const e = responseList[i];
if (getDavElementByTagName(e, 'resourcetype').getElementsByTagNameNS(nsCalDAV, 'calendar').length) {
const displayNameElement = getElementsInNamespaces(e, 'displayname')[0];
const displayName = displayNameElement ? displayNameElement.textContent.trim() : '';
const hrefElement = getElementsInNamespaces(e, 'href')[0];
const href = hrefElement ? hrefElement.textContent.trim() : '';
const calendarColorElement = getElementsInNamespaces(e, 'calendar-color')[0];
const calendarColor = calendarColorElement ? calendarColorElement.textContent.trim() : '#000000';
// Create an object to hold calendar data
const calendarData = {
displayName,
href,
calendarColor
};
// Call the function to create and append the list item
this.createCalendarListItem(calendarData, this.tree);
}
}
})
.catch(err => console.error(err));
}
onHide() {
this.fResolve(this.select);
}
/*
beforeShow() {} // Happens before showModal()
onShow() {} // Happens after showModal()
afterShow() {} // Happens after showModal() animation transitionend
onHide() {} // Happens before animation transitionend
afterHide() {} // Happens after animation transitionend
close() {}
*/
}
rl.nextcloud = {
selectCalendar: () =>
new Promise(resolve => {
NextcloudCalendarsPopupView.showModal([
href => resolve(href),
]);
}),
calendarPut: (path, event) => {
davFetch('calendars', path + '/' + event.UID + '.ics', {
method: 'PUT',
headers: {
'Content-Type': 'text/calendar'
},
// Validation error in iCalendar: A calendar object on a CalDAV server MUST NOT have a METHOD property.
body: event.rawText
.replace('METHOD:', 'X-METHOD:')
// https://github.com/nextcloud/calendar/issues/4684
.replace('ATTENDEE:', 'X-ATTENDEE:')
.replace('ORGANIZER:', 'X-ORGANIZER:')
.replace(/RSVP=TRUE/g, 'RSVP=FALSE')
.replace(/\r?\n/g, '\r\n')
})
.then(response => {
if (201 == response.status) {
// Created
} else if (204 == response.status) {
// Not modified
} else {
// response.text().then(text => console.error({status:response.status, body:text}));
Promise.reject(new Error({ response }));
}
});
},
selectFolder: () =>
new Promise(resolve => {
NextcloudFilesPopupView.showModal([
false,
folder => resolve(folder),
]);
}),
selectFiles: () =>
new Promise(resolve => {
NextcloudFilesPopupView.showModal([
true,
files => resolve(files),
]);
})
};
})(window.rl);
function getElementsInNamespaces(xmlDocument, tagName) {
const namespaces = {
d: 'DAV:',
x1: 'http://apple.com/ns/ical/',
};
const results = [];
for (const prefix in namespaces) {
const namespaceURI = namespaces[prefix];
const elements = xmlDocument.getElementsByTagNameNS(namespaceURI, tagName);
for (const element of elements) {
results.push(element);
}
}
return results;
}