Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
erp5
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Yoji Takeuchi
erp5
Commits
7d40af9c
Commit
7d40af9c
authored
Dec 28, 2015
by
Romain Courteaud
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[erp5_web_renderjs_ui] Update to jIO 3.7.0
parent
c528788e
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
789 additions
and
419 deletions
+789
-419
bt5/erp5_web_renderjs_ui/PathTemplateItem/web_page_module/rjs_jio_js.xml
...nderjs_ui/PathTemplateItem/web_page_module/rjs_jio_js.xml
+789
-419
No files found.
bt5/erp5_web_renderjs_ui/PathTemplateItem/web_page_module/rjs_jio_js.xml
View file @
7d40af9c
...
@@ -7584,115 +7584,6 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
...
@@ -7584,115 +7584,6 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
jIO.addStorage(\'memory\',
MemoryStorage);\n
jIO.addStorage(\'memory\',
MemoryStorage);\n
\n
\n
}(jIO));\n
}(jIO));\n
;/*\n
*
Copyright
2013,
Nexedi
SA\n
*
Released
under
the
LGPL
license.\n
*
http://www.gnu.org/licenses/lgpl.html\n
*/\n
\n
/*jslint
nomen:
true*/\n
/*global
jIO,
sessionStorage,
localStorage,
RSVP
*/\n
\n
/**\n
*
JIO
Local
Storage.
Type =
\'local\'.\n
*
Local
browser
"database"
storage.\n
*\n
*
Storage
Description:\n
*\n
*
{\n
*
"type":
"local",\n
*
"sessiononly":
false\n
*
}\n
*\n
*
@class
LocalStorage\n
*/\n
\n
(function
(jIO,
sessionStorage,
localStorage,
RSVP)
{\n
"use
strict";\n
\n
function
LocalStorage(spec)
{\n
if
(
spec.sessiononly =
==
true)
{\n
this._storage =
sessionStorage;\n
}
else
{\n
this._storage =
localStorage;\n
}\n
}\n
\n
function
restrictDocumentId(id)
{\n
if
(id
!==
"/")
{\n
throw
new
jIO.util.jIOError("id
"
+
id
+
"
is
forbidden
(!==
/)",\n
400);\n
}\n
}\n
\n
LocalStorage.prototype.get =
function
(id)
{\n
restrictDocumentId(id);\n
return
{};\n
};\n
\n
LocalStorage.prototype.allAttachments =
function
(id)
{\n
restrictDocumentId(id);\n
\n
var
attachments =
{},\n
key;\n
\n
for
(key
in
this._storage)
{\n
if
(this._storage.hasOwnProperty(key))
{\n
attachments[key]
=
{};\n
}\n
}\n
return
attachments;\n
};\n
\n
LocalStorage.prototype.getAttachment =
function
(id,
name)
{\n
restrictDocumentId(id);\n
\n
var
textstring =
this._storage.getItem(name);\n
\n
if
(
textstring =
==
null)
{\n
throw
new
jIO.util.jIOError(\n
"Cannot
find
attachment
"
+
name,\n
404\n
);\n
}\n
return
jIO.util.dataURItoBlob(textstring);\n
};\n
\n
LocalStorage.prototype.putAttachment =
function
(id,
name,
blob)
{\n
var
context =
this;\n
restrictDocumentId(id);\n
\n
//
the
document
already
exists\n
//
download
data\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
return
jIO.util.readBlobAsDataURL(blob);\n
})\n
.push(function
(e)
{\n
context._storage.setItem(name,
e.target.result);\n
});\n
};\n
\n
LocalStorage.prototype.removeAttachment =
function
(id,
name)
{\n
restrictDocumentId(id);\n
return
this._storage.removeItem(name);\n
};\n
\n
\n
LocalStorage.prototype.hasCapacity =
function
(name)
{\n
return
(
name =
==
"list");\n
};\n
\n
LocalStorage.prototype.buildQuery =
function
()
{\n
return
[{\n
id:
"/",\n
value:
{}\n
}];\n
};\n
\n
jIO.addStorage(\'local\',
LocalStorage);\n
\n
}(jIO,
sessionStorage,
localStorage,
RSVP));\n
;/*jslint
nomen:
true*/\n
;/*jslint
nomen:
true*/\n
/*global
RSVP,
Blob,
LZString,
DOMException*/\n
/*global
RSVP,
Blob,
LZString,
DOMException*/\n
(function
(RSVP,
Blob,
LZString,
DOMException)
{\n
(function
(RSVP,
Blob,
LZString,
DOMException)
{\n
...
@@ -7808,311 +7699,122 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
...
@@ -7808,311 +7699,122 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
jIO.addStorage(\'zip\',
ZipStorage);\n
jIO.addStorage(\'zip\',
ZipStorage);\n
}(RSVP,
Blob,
LZString,
DOMException));\n
}(RSVP,
Blob,
LZString,
DOMException));\n
;/*\n
;/*\n
*
Copyright
201
5
,
Nexedi
SA\n
*
Copyright
201
3
,
Nexedi
SA\n
*
Released
under
the
LGPL
license.\n
*
Released
under
the
LGPL
license.\n
*
http://www.gnu.org/licenses/lgpl.html\n
*
http://www.gnu.org/licenses/lgpl.html\n
*/\n
*/\n
\n
/**\n
*
JIO
Dropbox
Storage.
Type =
"dropbox"
.\n
*
Dropbox
"database"
storage.\n
*/\n
/*global
Blob,
jIO,
RSVP,
UriTemplate*/\n
/*jslint
nomen:
true*/\n
/*jslint
nomen:
true*/\n
/*global
jIO,
RSVP,
DOMException,
Blob,
crypto,
Uint8Array,
ArrayBuffer*/\n
\n
\n
(function
(jIO,
RSVP,
DOMException,
Blob,
crypto,
Uint8Array,
ArrayBuffer
)
{\n
(function
(jIO,
RSVP,
Blob,
UriTemplate
)
{\n
"use
strict";\n
"use
strict";\n
var
UPLOAD_URL =
"https://content.dropboxapi.com/1/files_put/"
+\n
"{+root}{+id}{+name}{?access_token}",\n
upload_template =
UriTemplate.parse(UPLOAD_URL),\n
CREATE_DIR_URL =
"https://api.dropboxapi.com/1/fileops/create_folder"
+\n
"{?access_token,root,path}",\n
create_dir_template =
UriTemplate.parse(CREATE_DIR_URL),\n
REMOVE_URL =
"https://api.dropboxapi.com/1/fileops/delete/"
+\n
"{?access_token,root,path}",\n
remote_template =
UriTemplate.parse(REMOVE_URL),\n
GET_URL =
"https://content.dropboxapi.com/1/files"
+\n
"{/root,id}{+name}{?access_token}",\n
get_template =
UriTemplate.parse(GET_URL),\n
//
LIST_URL =
\'https://api.dropboxapi.com/1/metadata/sandbox/\';\n
METADATA_URL =
"https://api.dropboxapi.com/1/metadata"
+\n
"{/root}{+id}{?access_token}",\n
metadata_template =
UriTemplate.parse(METADATA_URL);\n
\n
\n
function
restrictDocumentId(id)
{\n
if
(id.indexOf("/")
!==
0)
{\n
throw
new
jIO.util.jIOError("id
"
+
id
+
"
is
forbidden
(no
begin
/)",\n
400);\n
}\n
if
(id.lastIndexOf("/")
!==
(id.length
-
1))
{\n
throw
new
jIO.util.jIOError("id
"
+
id
+
"
is
forbidden
(no
end
/)",\n
400);\n
}\n
return
id;\n
}\n
\n
\n
//
you
the
cryptography
system
used
by
this
storage
is
AES-GCM.\n
function
restrictAttachmentId(id)
{\n
//
here
is
an
example
of
how
to
generate
a
key
to
the
json
format.\n
if
(id.indexOf("/")
!==
-1)
{\n
\n
throw
new
jIO.util.jIOError("attachment
"
+
id
+
"
is
forbidden",\n
//
var
key,\n
400);\n
//
jsonKey;\n
}\n
//
crypto.subtle.generateKey({name:
"AES-GCM",length:
256},\n
}\n
//
(true),
["encrypt",
"decrypt"])\n
//
.then(function(res){
key =
res;});\n
//\n
//
window.crypto.subtle.exportKey("jwk",
key)\n
//
.then(function(res){
jsonKey =
val})\n
//\n
//var
storage =
jIO.createJIO({type:
"crypt",
key:
jsonKey,\n
//
sub_storage:
{...}});\n
\n
//
find
more
informations
about
this
cryptography
system
on\n
//
https://github.com/diafygi/webcrypto-examples#aes-gcm\n
\n
\n
/**\n
/**\n
*
The
JIO
Cryptography
Storage
extension\n
*
The
JIO
Dropbox
Storage
extension\n
*\n
*\n
*
@class
Crypt
Storage\n
*
@class
Dropbox
Storage\n
*
@constructor\n
*
@constructor\n
*/\n
*/\n
\n
function
DropboxStorage(spec)
{\n
var
MIME_TYPE =
"application/x-jio-aes-gcm-encryption"
;\n
if
(typeof
spec.access_token
!==
\'string\'
||
!spec.access_token)
{\n
\n
throw
new
TypeError("Access
Token\'
must
be
a
string
"
+\n
function
CryptStorage(spec)
{\n
"which
contains
more
than
one
character.");\n
this._key =
spec.key;\n
}\n
this._jsonKey =
true;\n
if
(typeof
spec.root
!==
\'string\'
||
!spec.root
||\n
this._sub_storage =
jIO.createJIO(spec.sub_storage);\n
(spec.root
!==
"dropbox"
&&
spec.root
!==
"sandbox"))
{\n
throw
new
TypeError("root
must
be
\'dropbox\'
or
\'sandbox\'");\n
}\n
this._access_token =
spec.access_token;\n
this._root =
spec.root;\n
}\n
}\n
\n
\n
function
convertKey(that)
{\n
DropboxStorage.prototype.put =
function
(id,
param)
{\n
var
that =
this;\n
id =
restrictDocumentId(id);\n
if
(Object.getOwnPropertyNames(param).length
>
0) {\n
// Reject if param has some properties\n
throw new jIO.util.jIOError("Can not store properties: " +\n
Object.getOwnPropertyNames(param), 400);\n
}\n
return new RSVP.Queue()\n
return new RSVP.Queue()\n
.push(function () {\n
.push(function () {\n
return
crypto.subtle.importKey("jwk",
that._key,\n
return jIO.util.ajax({\n
"AES-GCM",
false,\n
type: "POST",\n
["encrypt",
"decrypt"]);\n
url: create_dir_template.expand({\n
access_token: that._access_token,\n
root: that._root,\n
path: id\n
})\n
});\n
})\n
})\n
.push(function
(res)
{\n
.push(undefined, function (err) {\n
that._key =
res;\n
if ((err.target !== undefined)
&&
\n
that._jsonKey =
false;\n
(err.target.status === 405)) {\n
return;\n
// Directory already exists, no need to fail\n
},
function
()
{\n
return;\n
throw
new
TypeError(\n
}\n
"\'key\'
must
be
a
CryptoKey
to
JSON
Web
Key
format"\n
throw err;\n
);\n
});\n
});\n
}\n
\n
CryptStorage.prototype.get =
function
()
{\n
return
this._sub_storage.get.apply(this._sub_storage,\n
arguments);\n
};\n
};\n
\n
\n
CryptStorage.prototype.post =
function
()
{\n
DropboxStorage.prototype.remove = function (id) {\n
return
this._sub_storage.post.apply(this._sub_storage,\n
id = restrictDocumentId(id);\n
arguments);\n
return jIO.util.ajax({\n
type: "POST",\n
url: remote_template.expand({\n
access_token: this._access_token,\n
root: this._root,\n
path: id\n
})\n
});\n
};\n
};\n
\n
\n
CryptStorage.prototype.put =
function
()
{\n
DropboxStorage.prototype.get = function (id) {\n
return
this._sub_storage.put.apply(this._sub_storage,\n
var that = this;\n
arguments);\n
};\n
\n
\n
CryptStorage.prototype.remove =
function
()
{\n
if (id === "/") {\n
return
this._sub_storage.remove.apply(this._sub_storage,\n
return {};\n
arguments);\n
}\n
};\n
id = restrictDocumentId(id);\n
\n
CryptStorage.prototype.hasCapacity =
function
()
{\n
return
this._sub_storage.hasCapacity.apply(this._sub_storage,\n
arguments);\n
};\n
\n
CryptStorage.prototype.buildQuery =
function
()
{\n
return
this._sub_storage.buildQuery.apply(this._sub_storage,\n
arguments);\n
};\n
\n
\n
CryptStorage.prototype.putAttachment =
function
(id,
name,
blob)
{\n
var
initializaton_vector =
crypto.getRandomValues(new
Uint8Array(12)),\n
that =
this;\n
\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
if
(
that._jsonKey =
==
true)
{\n
return
convertKey(that);\n
}\n
return;\n
})\n
.push(function
()
{\n
return
jIO.util.readBlobAsDataURL(blob);\n
})\n
.push(function
(dataURL)
{\n
//string-
>
arraybuffer\n
var strLen = dataURL.currentTarget.result.length,\n
buf = new ArrayBuffer(strLen),\n
bufView = new Uint8Array(buf),\n
i;\n
\n
dataURL = dataURL.currentTarget.result;\n
for (i = 0; i
< strLen
;
i
+=
1)
{\n
bufView[i]
=
dataURL.charCodeAt(i);\n
}\n
return
crypto.subtle.encrypt({\n
name
:
"AES-GCM",\n
iv
:
initializaton_vector\n
},\n
that._key,
buf);\n
})\n
.push(function
(coded)
{\n
var
blob =
new
Blob([initializaton_vector,
coded],
{type:
MIME_TYPE});\n
return
that._sub_storage.putAttachment(id,
name,
blob);\n
});\n
};\n
\n
CryptStorage.prototype.getAttachment =
function
(id,
name)
{\n
var
that =
this;\n
\n
return
that._sub_storage.getAttachment(id,
name)\n
.push(function
(blob)
{\n
if
(blob.type
!==
MIME_TYPE)
{\n
return
blob;\n
}\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
if
(
that._jsonKey =
==
true)
{\n
return
convertKey(that);\n
}\n
return;\n
})\n
.push(function
()
{\n
return
jIO.util.readBlobAsArrayBuffer(blob);\n
})\n
.push(function
(coded)
{\n
var
initializaton_vector;\n
\n
coded =
coded.currentTarget.result;\n
initializaton_vector =
new
Uint8Array(coded.slice(0,
12));\n
return
crypto.subtle.decrypt({\n
name
:
"AES-GCM",\n
iv
:
initializaton_vector\n
},\n
that._key,
coded.slice(12));\n
})\n
.push(function
(arr)
{\n
//arraybuffer-
>
string\n
arr = String.fromCharCode.apply(null, new Uint8Array(arr));\n
try {\n
return jIO.util.dataURItoBlob(arr);\n
} catch (error) {\n
if (error instanceof DOMException) {\n
return blob;\n
}\n
throw error;\n
}\n
}, function () { return blob; });\n
});\n
};\n
\n
CryptStorage.prototype.removeAttachment = function () {\n
return this._sub_storage.removeAttachment.apply(this._sub_storage,\n
arguments);\n
};\n
\n
CryptStorage.prototype.allAttachments = function () {\n
return this._sub_storage.allAttachments.apply(this._sub_storage,\n
arguments);\n
};\n
\n
jIO.addStorage(\'crypt\', CryptStorage);\n
\n
}(jIO, RSVP, DOMException, Blob, crypto, Uint8Array, ArrayBuffer));\n
;/*\n
* Copyright 2013, Nexedi SA\n
* Released under the LGPL license.\n
* http://www.gnu.org/licenses/lgpl.html\n
*/\n
/**\n
* JIO Dropbox Storage. Type = "dropbox".\n
* Dropbox "database" storage.\n
*/\n
/*global Blob, jIO, RSVP, UriTemplate*/\n
/*jslint nomen: true*/\n
\n
(function (jIO, RSVP, Blob, UriTemplate) {\n
"use strict";\n
var UPLOAD_URL = "https://content.dropboxapi.com/1/files_put/" +\n
"{+root}{+id}{+name}{?access_token}",\n
upload_template = UriTemplate.parse(UPLOAD_URL),\n
CREATE_DIR_URL = "https://api.dropboxapi.com/1/fileops/create_folder" +\n
"{?access_token,root,path}",\n
create_dir_template = UriTemplate.parse(CREATE_DIR_URL),\n
REMOVE_URL = "https://api.dropboxapi.com/1/fileops/delete/" +\n
"{?access_token,root,path}",\n
remote_template = UriTemplate.parse(REMOVE_URL),\n
GET_URL = "https://content.dropboxapi.com/1/files" +\n
"{/root,id}{+name}{?access_token}",\n
get_template = UriTemplate.parse(GET_URL),\n
//LIST_URL = \'https://api.dropboxapi.com/1/metadata/sandbox/\';\n
METADATA_URL = "https://api.dropboxapi.com/1/metadata" +\n
"{/root}{+id}{?access_token}",\n
metadata_template = UriTemplate.parse(METADATA_URL);\n
\n
function restrictDocumentId(id) {\n
if (id.indexOf("/") !== 0) {\n
throw new jIO.util.jIOError("id " + id + " is forbidden (no begin /)",\n
400);\n
}\n
if (id.lastIndexOf("/") !== (id.length - 1)) {\n
throw new jIO.util.jIOError("id " + id + " is forbidden (no end /)",\n
400);\n
}\n
return id;\n
}\n
\n
function restrictAttachmentId(id) {\n
if (id.indexOf("/") !== -1) {\n
throw new jIO.util.jIOError("attachment " + id + " is forbidden",\n
400);\n
}\n
}\n
\n
/**\n
* The JIO Dropbox Storage extension\n
*\n
* @class DropboxStorage\n
* @constructor\n
*/\n
function DropboxStorage(spec) {\n
if (typeof spec.access_token !== \'string\' || !spec.access_token) {\n
throw new TypeError("Access Token\' must be a string " +\n
"which contains more than one character.");\n
}\n
if (typeof spec.root !== \'string\' || !spec.root ||\n
(spec.root !== "dropbox"
&&
spec.root !== "sandbox")) {\n
throw new TypeError("root must be \'dropbox\' or \'sandbox\'");\n
}\n
this._access_token = spec.access_token;\n
this._root = spec.root;\n
}\n
\n
DropboxStorage.prototype.put = function (id, param) {\n
var that = this;\n
id = restrictDocumentId(id);\n
if (Object.getOwnPropertyNames(param).length > 0) {\n
// Reject if param has some properties\n
throw new jIO.util.jIOError("Can not store properties: " +\n
Object.getOwnPropertyNames(param), 400);\n
}\n
return new RSVP.Queue()\n
.push(function () {\n
return jIO.util.ajax({\n
type: "POST",\n
url: create_dir_template.expand({\n
access_token: that._access_token,\n
root: that._root,\n
path: id\n
})\n
});\n
})\n
.push(undefined, function (err) {\n
if ((err.target !== undefined)
&&
\n
(err.target.status === 405)) {\n
// Directory already exists, no need to fail\n
return;\n
}\n
throw err;\n
});\n
};\n
\n
DropboxStorage.prototype.remove = function (id) {\n
id = restrictDocumentId(id);\n
return jIO.util.ajax({\n
type: "POST",\n
url: remote_template.expand({\n
access_token: this._access_token,\n
root: this._root,\n
path: id\n
})\n
});\n
};\n
\n
DropboxStorage.prototype.get = function (id) {\n
var that = this;\n
\n
if (id === "/") {\n
return {};\n
}\n
id = restrictDocumentId(id);\n
\n
\n
return new RSVP.Queue()\n
return new RSVP.Queue()\n
.push(function () {\n
.push(function () {\n
...
@@ -8297,6 +7999,13 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
...
@@ -8297,6 +7999,13 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
}\n
}\n
options.headers.Authorization = storage._authorization;\n
options.headers.Authorization = storage._authorization;\n
}\n
}\n
\n
if (storage._with_credentials !== undefined) {\n
if (options.xhrFields === undefined) {\n
options.xhrFields = {};\n
}\n
options.xhrFields.withCredentials = storage._with_credentials;\n
}\n
// if (start !== undefined) {\n
// if (start !== undefined) {\n
// if (end !== undefined) {\n
// if (end !== undefined) {\n
// headers.Range = "bytes=" + start + "-" + end;\n
// headers.Range = "bytes=" + start + "-" + end;\n
...
@@ -8344,7 +8053,7 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
...
@@ -8344,7 +8053,7 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
if (typeof spec.basic_login === \'string\') {\n
if (typeof spec.basic_login === \'string\') {\n
this._authorization = "Basic " + spec.basic_login;\n
this._authorization = "Basic " + spec.basic_login;\n
}\n
}\n
\n
this._with_credentials = spec.with_credentials;
\n
}\n
}\n
\n
\n
DavStorage.prototype.put = function (id, param) {\n
DavStorage.prototype.put = function (id, param) {\n
...
@@ -10279,43 +9988,152 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
...
@@ -10279,43 +9988,152 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
\n
\n
}(jIO,
Blob,
atob,
btoa,
RSVP));\n
}(jIO,
Blob,
atob,
btoa,
RSVP));\n
;/*\n
;/*\n
*
Copyright
201
4
,
Nexedi
SA\n
*
Copyright
201
3
,
Nexedi
SA\n
*
Released
under
the
LGPL
license.\n
*
Released
under
the
LGPL
license.\n
*
http://www.gnu.org/licenses/lgpl.html\n
*
http://www.gnu.org/licenses/lgpl.html\n
*/\n
*/\n
\n
\n
/*jslint
nomen:
true*/\n
/*global
jIO,
sessionStorage,
localStorage,
RSVP
*/\n
\n
/**\n
/**\n
*
JIO
Indexed
Database
Storage.\n
*
JIO
Local
Storage.
Type =
\'local\'.\n
*\n
*
Local
browser
"database"
storage.\n
*
A
local
browser
"database"
storage
greatly
more
powerful
than
localStorage.\n
*\n
*
Description:\n
*\n
*
{\n
*
"type":
"indexeddb",\n
*
"database":
<string
>
\n
* }\n
*\n
*\n
* The database name will be prefixed by "jio:", so if the database property is\n
*
Storage
Description:\n
* "hello", then you can manually reach this database with\n
* `indexedDB.open("jio:hello");`. (Or\n
* `indexedDB.deleteDatabase("jio:hello");`.)\n
*\n
*\n
* For more informations:\n
*
{\n
*
"type":
"local",\n
*
"sessiononly":
false\n
*
}\n
*\n
*\n
* - http://www.w3.org/TR/IndexedDB/\n
*
@class
LocalStorage\n
* - https://developer.mozilla.org/en-US/docs/IndexedDB/Using_IndexedDB\n
*/\n
*/\n
\n
\n
/*jslint nomen: true */\n
(function
(jIO,
sessionStorage,
localStorage,
RSVP)
{\n
/*global indexedDB, jIO, RSVP, Blob, Math, IDBKeyRange*/\n
\n
(function (indexedDB, jIO, RSVP, Blob, Math, IDBKeyRange) {\n
"use
strict";\n
"use
strict";\n
\n
\n
// Read only as changing it can lead to data corruption\n
function
LocalStorage(spec)
{\n
var UNITE = 2000000;\n
if
(
spec.sessiononly =
==
true)
{\n
\n
this._storage =
sessionStorage;\n
}
else
{\n
this._storage =
localStorage;\n
}\n
}\n
\n
function
restrictDocumentId(id)
{\n
if
(id
!==
"/")
{\n
throw
new
jIO.util.jIOError("id
"
+
id
+
"
is
forbidden
(!==
/)",\n
400);\n
}\n
}\n
\n
LocalStorage.prototype.get =
function
(id)
{\n
restrictDocumentId(id);\n
return
{};\n
};\n
\n
LocalStorage.prototype.allAttachments =
function
(id)
{\n
restrictDocumentId(id);\n
\n
var
attachments =
{},\n
key;\n
\n
for
(key
in
this._storage)
{\n
if
(this._storage.hasOwnProperty(key))
{\n
attachments[key]
=
{};\n
}\n
}\n
return
attachments;\n
};\n
\n
LocalStorage.prototype.getAttachment =
function
(id,
name)
{\n
restrictDocumentId(id);\n
\n
var
textstring =
this._storage.getItem(name);\n
\n
if
(
textstring =
==
null)
{\n
throw
new
jIO.util.jIOError(\n
"Cannot
find
attachment
"
+
name,\n
404\n
);\n
}\n
return
jIO.util.dataURItoBlob(textstring);\n
};\n
\n
LocalStorage.prototype.putAttachment =
function
(id,
name,
blob)
{\n
var
context =
this;\n
restrictDocumentId(id);\n
\n
//
the
document
already
exists\n
//
download
data\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
return
jIO.util.readBlobAsDataURL(blob);\n
})\n
.push(function
(e)
{\n
context._storage.setItem(name,
e.target.result);\n
});\n
};\n
\n
LocalStorage.prototype.removeAttachment =
function
(id,
name)
{\n
restrictDocumentId(id);\n
return
this._storage.removeItem(name);\n
};\n
\n
\n
LocalStorage.prototype.hasCapacity =
function
(name)
{\n
return
(
name =
==
"list");\n
};\n
\n
LocalStorage.prototype.buildQuery =
function
()
{\n
return
[{\n
id:
"/",\n
value:
{}\n
}];\n
};\n
\n
jIO.addStorage(\'local\',
LocalStorage);\n
\n
}(jIO,
sessionStorage,
localStorage,
RSVP));\n
;/*\n
*
Copyright
2014,
Nexedi
SA\n
*
Released
under
the
LGPL
license.\n
*
http://www.gnu.org/licenses/lgpl.html\n
*/\n
\n
/**\n
*
JIO
Indexed
Database
Storage.\n
*\n
*
A
local
browser
"database"
storage
greatly
more
powerful
than
localStorage.\n
*\n
*
Description:\n
*\n
*
{\n
*
"type":
"indexeddb",\n
*
"database":
<string
>
\n
* }\n
*\n
* The database name will be prefixed by "jio:", so if the database property is\n
* "hello", then you can manually reach this database with\n
* `indexedDB.open("jio:hello");`. (Or\n
* `indexedDB.deleteDatabase("jio:hello");`.)\n
*\n
* For more informations:\n
*\n
* - http://www.w3.org/TR/IndexedDB/\n
* - https://developer.mozilla.org/en-US/docs/IndexedDB/Using_IndexedDB\n
*/\n
\n
/*jslint nomen: true */\n
/*global indexedDB, jIO, RSVP, Blob, Math, IDBKeyRange*/\n
\n
(function (indexedDB, jIO, RSVP, Blob, Math, IDBKeyRange) {\n
"use strict";\n
\n
// Read only as changing it can lead to data corruption\n
var UNITE = 2000000;\n
\n
function IndexedDBStorage(description) {\n
function IndexedDBStorage(description) {\n
if (typeof description.database !== "string" ||\n
if (typeof description.database !== "string" ||\n
description.database === "") {\n
description.database === "") {\n
...
@@ -10730,7 +10548,559 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
...
@@ -10730,7 +10548,559 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
};\n
};\n
\n
\n
jIO.addStorage("indexeddb",
IndexedDBStorage);\n
jIO.addStorage("indexeddb",
IndexedDBStorage);\n
}(indexedDB,
jIO,
RSVP,
Blob,
Math,
IDBKeyRange));
}(indexedDB,
jIO,
RSVP,
Blob,
Math,
IDBKeyRange));\n
;/*\n
*
Copyright
2015,
Nexedi
SA\n
*
Released
under
the
LGPL
license.\n
*
http://www.gnu.org/licenses/lgpl.html\n
*/\n
\n
/*jslint
nomen:
true*/\n
/*global
jIO,
RSVP,
DOMException,
Blob,
crypto,
Uint8Array,
ArrayBuffer*/\n
\n
(function
(jIO,
RSVP,
DOMException,
Blob,
crypto,
Uint8Array,
ArrayBuffer)
{\n
"use
strict";\n
\n
\n
//
you
the
cryptography
system
used
by
this
storage
is
AES-GCM.\n
//
here
is
an
example
of
how
to
generate
a
key
to
the
json
format.\n
\n
//
var
key,\n
//
jsonKey;\n
//
crypto.subtle.generateKey({name:
"AES-GCM",length:
256},\n
//
(true),
["encrypt",
"decrypt"])\n
//
.then(function(res){
key =
res;});\n
//\n
//
window.crypto.subtle.exportKey("jwk",
key)\n
//
.then(function(res){
jsonKey =
val})\n
//\n
//var
storage =
jIO.createJIO({type:
"crypt",
key:
jsonKey,\n
//
sub_storage:
{...}});\n
\n
//
find
more
informations
about
this
cryptography
system
on\n
//
https://github.com/diafygi/webcrypto-examples#aes-gcm\n
\n
/**\n
*
The
JIO
Cryptography
Storage
extension\n
*\n
*
@class
CryptStorage\n
*
@constructor\n
*/\n
\n
var
MIME_TYPE =
"application/x-jio-aes-gcm-encryption"
;\n
\n
function
CryptStorage(spec)
{\n
this._key =
spec.key;\n
this._jsonKey =
true;\n
this._sub_storage =
jIO.createJIO(spec.sub_storage);\n
}\n
\n
function
convertKey(that)
{\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
return
crypto.subtle.importKey("jwk",
that._key,\n
"AES-GCM",
false,\n
["encrypt",
"decrypt"]);\n
})\n
.push(function
(res)
{\n
that._key =
res;\n
that._jsonKey =
false;\n
return;\n
});\n
}\n
\n
CryptStorage.prototype.get =
function
()
{\n
return
this._sub_storage.get.apply(this._sub_storage,\n
arguments);\n
};\n
\n
CryptStorage.prototype.post =
function
()
{\n
return
this._sub_storage.post.apply(this._sub_storage,\n
arguments);\n
};\n
\n
CryptStorage.prototype.put =
function
()
{\n
return
this._sub_storage.put.apply(this._sub_storage,\n
arguments);\n
};\n
\n
CryptStorage.prototype.remove =
function
()
{\n
return
this._sub_storage.remove.apply(this._sub_storage,\n
arguments);\n
};\n
\n
CryptStorage.prototype.hasCapacity =
function
()
{\n
return
this._sub_storage.hasCapacity.apply(this._sub_storage,\n
arguments);\n
};\n
\n
CryptStorage.prototype.buildQuery =
function
()
{\n
return
this._sub_storage.buildQuery.apply(this._sub_storage,\n
arguments);\n
};\n
\n
\n
CryptStorage.prototype.putAttachment =
function
(id,
name,
blob)
{\n
var
initializaton_vector =
crypto.getRandomValues(new
Uint8Array(12)),\n
that =
this;\n
\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
if
(
that._jsonKey =
==
true)
{\n
return
convertKey(that);\n
}\n
return;\n
})\n
.push(function
()
{\n
return
jIO.util.readBlobAsDataURL(blob);\n
})\n
.push(function
(dataURL)
{\n
//string-
>
arraybuffer\n
var strLen = dataURL.currentTarget.result.length,\n
buf = new ArrayBuffer(strLen),\n
bufView = new Uint8Array(buf),\n
i;\n
\n
dataURL = dataURL.currentTarget.result;\n
for (i = 0; i
< strLen
;
i
+=
1)
{\n
bufView[i]
=
dataURL.charCodeAt(i);\n
}\n
return
crypto.subtle.encrypt({\n
name
:
"AES-GCM",\n
iv
:
initializaton_vector\n
},\n
that._key,
buf);\n
})\n
.push(function
(coded)
{\n
var
blob =
new
Blob([initializaton_vector,
coded],
{type:
MIME_TYPE});\n
return
that._sub_storage.putAttachment(id,
name,
blob);\n
});\n
};\n
\n
CryptStorage.prototype.getAttachment =
function
(id,
name)
{\n
var
that =
this;\n
\n
return
that._sub_storage.getAttachment(id,
name)\n
.push(function
(blob)
{\n
if
(blob.type
!==
MIME_TYPE)
{\n
return
blob;\n
}\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
if
(
that._jsonKey =
==
true)
{\n
return
convertKey(that);\n
}\n
return;\n
})\n
.push(function
()
{\n
return
jIO.util.readBlobAsArrayBuffer(blob);\n
})\n
.push(function
(coded)
{\n
var
initializaton_vector;\n
\n
coded =
coded.currentTarget.result;\n
initializaton_vector =
new
Uint8Array(coded.slice(0,
12));\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
return
crypto.subtle.decrypt({\n
name
:
"AES-GCM",\n
iv
:
initializaton_vector\n
},\n
that._key,
coded.slice(12));\n
})\n
.push(function
(arr)
{\n
//arraybuffer-
>
string\n
arr = String.fromCharCode.apply(null, new Uint8Array(arr));\n
return jIO.util.dataURItoBlob(arr);\n
})\n
.push(undefined, function (error) {\n
if (error instanceof DOMException) {\n
return blob;\n
}\n
throw error;\n
});\n
});\n
});\n
};\n
\n
CryptStorage.prototype.removeAttachment = function () {\n
return this._sub_storage.removeAttachment.apply(this._sub_storage,\n
arguments);\n
};\n
\n
CryptStorage.prototype.allAttachments = function () {\n
return this._sub_storage.allAttachments.apply(this._sub_storage,\n
arguments);\n
};\n
\n
jIO.addStorage(\'crypt\', CryptStorage);\n
\n
}(jIO, RSVP, DOMException, Blob, crypto, Uint8Array, ArrayBuffer));\n
;/*\n
* Copyright 2013, Nexedi SA\n
* Released under the LGPL license.\n
* http://www.gnu.org/licenses/lgpl.html\n
*/\n
/**\n
* JIO Websql Storage. Type = "websql".\n
* websql "database" storage.\n
*/\n
/*global Blob, jIO, RSVP, openDatabase*/\n
/*jslint nomen: true*/\n
\n
(function (jIO, RSVP, Blob, openDatabase) {\n
\n
"use strict";\n
\n
/**\n
* The JIO Websql Storage extension\n
*\n
* @class WebSQLStorage\n
* @constructor\n
*/\n
\n
function queueSql(db, query_list, argument_list) {\n
return new RSVP.Promise(function (resolve, reject) {\n
/*jslint unparam: true*/\n
db.transaction(function (tx) {\n
var len = query_list.length,\n
result_list = [],\n
i;\n
\n
function resolveTransaction(tx, result) {\n
result_list.push(result);\n
if (result_list.length === len) {\n
resolve(result_list);\n
}\n
}\n
function rejectTransaction(tx, error) {\n
reject(error);\n
return true;\n
}\n
for (i = 0; i
< len
;
i
+=
1)
{\n
tx.executeSql(query_list[i],
argument_list[i],
resolveTransaction,\n
rejectTransaction);\n
}\n
},
function
(tx,
error)
{\n
reject(error);\n
});\n
/*jslint
unparam:
false*/\n
});\n
}\n
\n
function
initDatabase(db)
{\n
var
query_list =
[\n
"CREATE
TABLE
IF
NOT
EXISTS
document"
+\n
"(id
VARCHAR
PRIMARY
KEY
NOT
NULL,
data
TEXT)",\n
"CREATE
TABLE
IF
NOT
EXISTS
attachment"
+\n
"(id
VARCHAR,
attachment
VARCHAR,
part
INT,
blob
TEXT)",\n
"CREATE
TRIGGER
IF
NOT
EXISTS
removeAttachment
"
+\n
"BEFORE
DELETE
ON
document
FOR
EACH
ROW
"
+\n
"BEGIN
DELETE
from
attachment
WHERE
id =
OLD.id;END;",\n
"CREATE
INDEX
IF
NOT
EXISTS
index_document
ON
document
(id);",\n
"CREATE
INDEX
IF
NOT
EXISTS
index_attachment
"
+\n
"ON
attachment
(id,
attachment);"\n
];\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
return
queueSql(db,
query_list,
[]);\n
});\n
}\n
\n
function
WebSQLStorage(spec)
{\n
if
(typeof
spec.database
!==
\'string\'
||
!spec.database)
{\n
throw
new
TypeError("database
must
be
a
string
"
+\n
"which
contains
more
than
one
character.");\n
}\n
this._database =
openDatabase("jio:"
+
spec.database,\n
\'1.0\',
\'\',
2
*
1024
*
1024);\n
if
(spec.blob_length
&&\n
(typeof
spec.blob_length
!==
"number"
||\n
spec.blob_length
<
20))
{\n
throw
new
TypeError("blob_len
parameter
must
be
a
number
>
= 20");\n
}\n
this._blob_length = spec.blob_length || 2000000;\n
this._init_db_promise = initDatabase(this._database);\n
}\n
\n
WebSQLStorage.prototype.put = function (id, param) {\n
var db = this._database,\n
that = this,\n
data_string = JSON.stringify(param);\n
\n
return new RSVP.Queue()\n
.push(function () {\n
return that._init_db_promise;\n
})\n
.push(function () {\n
return queueSql(db, ["INSERT OR REPLACE INTO " +\n
"document(id, data) VALUES(?,?)"],\n
[[id, data_string]]);\n
})\n
.push(function () {\n
return id;\n
});\n
};\n
\n
WebSQLStorage.prototype.remove = function (id) {\n
var db = this._database,\n
that = this;\n
\n
return new RSVP.Queue()\n
.push(function () {\n
return that._init_db_promise;\n
})\n
.push(function () {\n
return queueSql(db, ["DELETE FROM document WHERE id = ?"], [[id]]);\n
})\n
.push(function (result_list) {\n
if (result_list[0].rowsAffected === 0) {\n
throw new jIO.util.jIOError("Cannot find document", 404);\n
}\n
return id;\n
});\n
\n
};\n
\n
WebSQLStorage.prototype.get = function (id) {\n
var db = this._database,\n
that = this;\n
\n
return new RSVP.Queue()\n
.push(function () {\n
return that._init_db_promise;\n
})\n
.push(function () {\n
return queueSql(db, ["SELECT data FROM document WHERE id = ?"],\n
[[id]]);\n
})\n
.push(function (result_list) {\n
if (result_list[0].rows.length === 0) {\n
throw new jIO.util.jIOError("Cannot find document", 404);\n
}\n
return JSON.parse(result_list[0].rows[0].data);\n
});\n
};\n
\n
WebSQLStorage.prototype.allAttachments = function (id) {\n
var db = this._database,\n
that = this;\n
\n
return new RSVP.Queue()\n
.push(function () {\n
return that._init_db_promise;\n
})\n
.push(function () {\n
return queueSql(db, [\n
"SELECT id FROM document WHERE id = ?",\n
"SELECT DISTINCT attachment FROM attachment WHERE id = ?"\n
], [[id], [id]]);\n
})\n
.push(function (result_list) {\n
if (result_list[0].rows.length === 0) {\n
throw new jIO.util.jIOError("Cannot find document", 404);\n
}\n
\n
var len = result_list[1].rows.length,\n
obj = {},\n
i;\n
\n
for (i = 0; i
< len
;
i
+=
1)
{\n
obj[result_list[1].rows[i].attachment]
=
{};\n
}\n
return
obj;\n
});\n
};\n
\n
function
sendBlobPart(blob,
argument_list,
index,
queue)
{\n
queue.push(function
()
{\n
return
jIO.util.readBlobAsDataURL(blob);\n
})\n
.push(function
(strBlob)
{\n
argument_list[index
+
2].push(strBlob.currentTarget.result);\n
return;\n
});\n
}\n
\n
WebSQLStorage.prototype.putAttachment =
function
(id,
name,
blob)
{\n
var
db =
this._database,\n
that =
this,\n
part_size =
this._blob_length;\n
\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
return
that._init_db_promise;\n
})\n
.push(function
()
{\n
return
queueSql(db,
["SELECT
id
FROM
document
WHERE
id =
?"],
[[id]]);\n
})\n
.push(function
(result)
{\n
var
query_list =
[],\n
argument_list =
[],\n
blob_size =
blob.size,\n
queue =
new
RSVP.Queue(),\n
i,\n
index;\n
\n
if
(result[0]
.rows.length =
==
0)
{\n
throw
new
jIO.util.jIOError("Cannot
access
subdocument",
404);\n
}\n
query_list.push("DELETE
FROM
attachment
WHERE
id =
?
"
+\n
"AND
attachment =
?");\n
argument_list.push([id,
name]);\n
query_list.push("INSERT
INTO
attachment(id,
attachment,
part,
blob)"
+\n
"VALUES(?,
?,
?,
?)");\n
argument_list.push([id,
name,
-1,\n
blob.type
||
"application/octet-stream"]);\n
\n
for
(
i =
0,
index =
0;
i
<
blob_size;
i
+=
part_size,
index
+=
1)
{\n
query_list.push("INSERT
INTO
attachment(id,
attachment,
part,
blob)"
+\n
"VALUES(?,
?,
?,
?)");\n
argument_list.push([id,
name,
index]);\n
sendBlobPart(blob.slice(i,
i
+
part_size),
argument_list,
index,\n
queue);\n
}\n
queue.push(function
()
{\n
return
queueSql(db,
query_list,
argument_list);\n
});\n
return
queue;\n
});\n
};\n
\n
WebSQLStorage.prototype.getAttachment =
function
(id,
name,
options)
{\n
var
db =
this._database,\n
that =
this,\n
part_size =
this._blob_length,\n
start,\n
end,\n
start_index,\n
end_index;\n
\n
if
(
options =
==
undefined)
{
options =
{};
}\n
start =
options.start
||
0;\n
end =
options.end
||
-1;\n
\n
if
(start
<
0
||
(options.end
!==
undefined
&&
options.end
<
0))
{\n
throw
new
jIO.util.jIOError("_start
and
_end
must
be
positive",\n
400);\n
}\n
if
(start
>
end
&&
end !== -1) {\n
throw new jIO.util.jIOError("_start is greater than _end",\n
400);\n
}\n
\n
start_index = Math.floor(start / part_size);\n
if (start === 0) { start_index -= 1; }\n
end_index = Math.floor(end / part_size);\n
if (end % part_size === 0) {\n
end_index -= 1;\n
}\n
\n
return new RSVP.Queue()\n
.push(function () {\n
return that._init_db_promise;\n
})\n
.push(function () {\n
var command = "SELECT part, blob FROM attachment WHERE id = ? AND " +\n
"attachment = ? AND part >= ?",\n
argument_list = [id, name, start_index];\n
\n
if (end !== -1) {\n
command += " AND part
<
= ?";\n
argument_list.push(end_index);\n
}\n
return queueSql(db, [command], [argument_list]);\n
})\n
.push(function (response_list) {\n
var i,\n
response,\n
blob_array = [],\n
blob,\n
type;\n
\n
response = response_list[0].rows;\n
if (response.length === 0) {\n
throw new jIO.util.jIOError("Cannot find document", 404);\n
}\n
for (i = 0; i
< response.length
;
i
+=
1)
{\n
if
(response[i]
.part =
==
-1)
{\n
type =
response[i].blob;\n
start_index
+=
1;\n
}
else
{\n
blob_array.push(jIO.util.dataURItoBlob(response[i].blob));\n
}\n
}\n
if
((
start =
==
0)
&&
(
options.end =
==
undefined))
{\n
return
new
Blob(blob_array,
{type:
type});\n
}\n
blob =
new
Blob(blob_array,
{});\n
return
blob.slice(start
-
(start_index
*
part_size),\n
end =
==
-1
?
blob.size
:\n
end
-
(start_index
*
part_size),\n
"application/octet-stream");\n
});\n
};\n
\n
WebSQLStorage.prototype.removeAttachment =
function
(id,
name)
{\n
var
db =
this._database,\n
that =
this;\n
\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
return
that._init_db_promise;\n
})\n
.push(function
()
{\n
return
queueSql(db,
["DELETE
FROM
attachment
WHERE
"
+\n
"
id =
?
AND
attachment =
?"],
[[id,
name]]);\n
})\n
.push(function
(result)
{\n
if
(result[0]
.rowsAffected =
==
0)
{\n
throw
new
jIO.util.jIOError("Cannot
find
document",
404);\n
}\n
return
name;\n
});\n
};\n
\n
WebSQLStorage.prototype.hasCapacity =
function
(name)
{\n
return
(
name =
==
"list"
||
(
name =
==
"include"));\n
};\n
\n
WebSQLStorage.prototype.buildQuery =
function
(options)
{\n
var
db =
this._database,\n
that =
this,\n
query =
"SELECT id"
;\n
\n
return
new
RSVP.Queue()\n
.push(function
()
{\n
return
that._init_db_promise;\n
})\n
.push(function
()
{\n
if
(
options =
==
undefined)
{
options =
{};
}\n
if
(
options.include_docs =
==
true)
{\n
query
+=
",
data
AS
doc";\n
}\n
query
+=
"
FROM
document";\n
return
queueSql(db,
[query],
[[]]);\n
})\n
.push(function
(result)
{\n
var
array =
[],\n
len =
result[0].rows.length,\n
i;\n
\n
for
(
i =
0;
i
<
len;
i
+=
1)
{\n
array.push(result[0].rows[i]);\n
array[i]
.value =
{};\n
if
(array[i].doc
!==
undefined)
{\n
array[i]
.doc =
JSON.parse(array[i].doc);\n
}\n
}\n
return
array;\n
});\n
};\n
\n
jIO.addStorage(\'websql\',
WebSQLStorage);\n
\n
}(jIO,
RSVP,
Blob,
openDatabase));
]]
></string>
</value>
]]
></string>
</value>
</item>
</item>
...
@@ -10867,7 +11237,7 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
...
@@ -10867,7 +11237,7 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
</item>
</item>
<item>
<item>
<key>
<string>
serial
</string>
</key>
<key>
<string>
serial
</string>
</key>
<value>
<string>
947.
22494.14742.48810
</string>
</value>
<value>
<string>
947.
46978.10225.52394
</string>
</value>
</item>
</item>
<item>
<item>
<key>
<string>
state
</string>
</key>
<key>
<string>
state
</string>
</key>
...
@@ -10885,7 +11255,7 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
...
@@ -10885,7 +11255,7 @@ if ((error_count = __NODEJS_parse(string, error_offsets, error_lookaheads)) > 0)
</tuple>
</tuple>
<state>
<state>
<tuple>
<tuple>
<float>
14
49226044.59
</float>
<float>
14
51317168.97
</float>
<string>
UTC
</string>
<string>
UTC
</string>
</tuple>
</tuple>
</state>
</state>
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment