2017-06-22 20:44:38 +01:00
|
|
|
import moment from 'moment';
|
2017-06-24 19:06:28 +01:00
|
|
|
import { time } from 'lib/time-utils.js';
|
|
|
|
import { dirname, basename } from 'lib/path-utils.js';
|
|
|
|
import { OneDriveApi } from 'lib/onedrive-api.js';
|
2017-06-22 20:44:38 +01:00
|
|
|
|
|
|
|
class FileApiDriverOneDrive {
|
|
|
|
|
2017-07-06 19:29:09 +00:00
|
|
|
constructor(api) {
|
|
|
|
this.api_ = api;
|
2017-06-22 22:52:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
api() {
|
|
|
|
return this.api_;
|
2017-06-22 20:44:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
itemFilter_() {
|
|
|
|
return {
|
|
|
|
select: 'name,file,folder,fileSystemInfo',
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
makePath_(path) {
|
2017-06-22 22:52:27 +01:00
|
|
|
return path;
|
2017-06-22 20:44:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
makeItems_(odItems) {
|
|
|
|
let output = [];
|
|
|
|
for (let i = 0; i < odItems.length; i++) {
|
|
|
|
output.push(this.makeItem_(odItems[i]));
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
|
|
|
makeItem_(odItem) {
|
|
|
|
return {
|
|
|
|
path: odItem.name,
|
|
|
|
isDir: ('folder' in odItem),
|
|
|
|
created_time: moment(odItem.fileSystemInfo.createdDateTime, 'YYYY-MM-DDTHH:mm:ss.SSSZ').format('x'),
|
|
|
|
updated_time: moment(odItem.fileSystemInfo.lastModifiedDateTime, 'YYYY-MM-DDTHH:mm:ss.SSSZ').format('x'),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-06-23 23:19:59 +01:00
|
|
|
async statRaw_(path) {
|
2017-06-23 22:32:24 +01:00
|
|
|
let item = null;
|
2017-06-22 22:52:27 +01:00
|
|
|
try {
|
2017-06-23 22:32:24 +01:00
|
|
|
item = await this.api_.execJson('GET', this.makePath_(path), this.itemFilter_());
|
2017-06-22 22:52:27 +01:00
|
|
|
} catch (error) {
|
2017-06-29 18:03:16 +00:00
|
|
|
if (error.code == 'itemNotFound') return null;
|
2017-06-22 22:52:27 +01:00
|
|
|
throw error;
|
|
|
|
}
|
2017-06-23 23:19:59 +01:00
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
async stat(path) {
|
|
|
|
let item = await this.statRaw_(path);
|
|
|
|
if (!item) return null;
|
2017-06-22 20:44:38 +01:00
|
|
|
return this.makeItem_(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
async setTimestamp(path, timestamp) {
|
|
|
|
let body = {
|
|
|
|
fileSystemInfo: {
|
|
|
|
lastModifiedDateTime: moment.unix(timestamp / 1000).utc().format('YYYY-MM-DDTHH:mm:ss.SSS') + 'Z',
|
|
|
|
}
|
|
|
|
};
|
2017-06-23 23:19:59 +01:00
|
|
|
let item = await this.api_.execJson('PATCH', this.makePath_(path), null, body);
|
|
|
|
return this.makeItem_(item);
|
2017-06-22 20:44:38 +01:00
|
|
|
}
|
|
|
|
|
2017-06-29 18:03:16 +00:00
|
|
|
async list(path, options = null) {
|
|
|
|
let query = this.itemFilter_();
|
|
|
|
let url = this.makePath_(path) + ':/children';
|
|
|
|
|
|
|
|
if (options.context) {
|
|
|
|
query = null;
|
|
|
|
url = options.context;
|
|
|
|
}
|
|
|
|
|
|
|
|
let r = await this.api_.execJson('GET', url, query);
|
|
|
|
|
|
|
|
return {
|
|
|
|
hasMore: !!r['@odata.nextLink'],
|
|
|
|
items: this.makeItems_(r.value),
|
|
|
|
context: r["@odata.nextLink"],
|
|
|
|
}
|
2017-06-22 20:44:38 +01:00
|
|
|
}
|
|
|
|
|
2017-07-06 22:30:45 +01:00
|
|
|
async get(path, options = null) {
|
|
|
|
if (!options) options = {};
|
|
|
|
|
2017-06-22 22:52:27 +01:00
|
|
|
try {
|
2017-07-06 22:30:45 +01:00
|
|
|
if (options.target == 'file') {
|
|
|
|
let response = await this.api_.exec('GET', this.makePath_(path) + ':/content', null, null, options);
|
|
|
|
return response;
|
|
|
|
} else {
|
|
|
|
let content = await this.api_.execText('GET', this.makePath_(path) + ':/content');
|
|
|
|
return content;
|
|
|
|
}
|
2017-06-22 22:52:27 +01:00
|
|
|
} catch (error) {
|
2017-06-29 18:03:16 +00:00
|
|
|
if (error.code == 'itemNotFound') return null;
|
2017-06-22 22:52:27 +01:00
|
|
|
throw error;
|
|
|
|
}
|
2017-06-22 20:44:38 +01:00
|
|
|
}
|
|
|
|
|
2017-06-23 22:32:24 +01:00
|
|
|
async mkdir(path) {
|
|
|
|
let item = await this.stat(path);
|
|
|
|
if (item) return item;
|
|
|
|
|
|
|
|
let parentPath = dirname(path);
|
|
|
|
item = await this.api_.execJson('POST', this.makePath_(parentPath) + ':/children', this.itemFilter_(), {
|
|
|
|
name: basename(path),
|
|
|
|
folder: {},
|
|
|
|
});
|
|
|
|
|
|
|
|
return this.makeItem_(item);
|
2017-06-22 20:44:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
put(path, content) {
|
|
|
|
let options = {
|
|
|
|
headers: { 'Content-Type': 'text/plain' },
|
|
|
|
};
|
|
|
|
return this.api_.exec('PUT', this.makePath_(path) + ':/content', null, content, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(path) {
|
|
|
|
return this.api_.exec('DELETE', this.makePath_(path));
|
|
|
|
}
|
|
|
|
|
2017-06-23 22:32:24 +01:00
|
|
|
async move(oldPath, newPath) {
|
2017-07-12 23:32:08 +01:00
|
|
|
// Cannot work in an atomic way because if newPath already exist, the OneDrive API throw an error
|
|
|
|
// "An item with the same name already exists under the parent". Some posts suggest to use
|
|
|
|
// @name.conflictBehavior [0]but that doesn't seem to work. So until Microsoft fixes this
|
|
|
|
// it's not possible to do an atomic move.
|
|
|
|
//
|
|
|
|
// [0] https://stackoverflow.com/questions/29191091/onedrive-api-overwrite-on-move
|
|
|
|
throw new Error('NOT WORKING');
|
|
|
|
|
2017-06-23 23:19:59 +01:00
|
|
|
let previousItem = await this.statRaw_(oldPath);
|
|
|
|
|
2017-06-23 22:32:24 +01:00
|
|
|
let newDir = dirname(newPath);
|
|
|
|
let newName = basename(newPath);
|
|
|
|
|
2017-06-23 23:19:59 +01:00
|
|
|
// We don't want the modification date to change when we move the file so retrieve it
|
|
|
|
// now set it in the PATCH operation.
|
|
|
|
|
2017-06-23 22:32:24 +01:00
|
|
|
let item = await this.api_.execJson('PATCH', this.makePath_(oldPath), this.itemFilter_(), {
|
|
|
|
name: newName,
|
|
|
|
parentReference: { path: newDir },
|
2017-06-23 23:19:59 +01:00
|
|
|
fileSystemInfo: {
|
|
|
|
lastModifiedDateTime: previousItem.fileSystemInfo.lastModifiedDateTime,
|
|
|
|
},
|
2017-06-23 22:32:24 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
return this.makeItem_(item);
|
2017-06-22 20:44:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
format() {
|
|
|
|
throw new Error('Not implemented');
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
export { FileApiDriverOneDrive };
|