Skip to content

Commit 92fd07a

Browse files
colinfruitlidel
authored andcommittedDec 3, 2019
feat: import files to MFS (#810)
* upload files to MFS * format directory path properly on upload * move open web UI call out of file upload block * reword upload to import in UI messages * move default import directory to preferences screen * Ensure imported files have the same CID when imported from Web UI or ipfs-companion This has been done using adding them to IPFS first (ipfs.add) first and then using the ipfs files copy api (ipfs.files.cp) to move the files into the desired directory. * open resource at gateway if using embedded node * add date symbols to import directory * add import options to import screen * fix: only display openViaWebUI option if opening via web UI is possible * fix: remove wrapping directory from files API import * upload => import in MFS import variables * trim double slashes from import directory * add file import section on options page * move preloadAtPublicGateway option to file import section * preload files when opening imported files in web UI * test quick-upload formatImportDirectory * add openViaWebUI option to file import form * Remove low-level pin when using ipfs.add * Import to MFS from context menu This commit required abstracting the ipfs import functionality into a separate file to avoid a circular dependency between quick-upload and ipfs-companion, and continue to enable testing functionality.
1 parent 36a1a7e commit 92fd07a

File tree

10 files changed

+354
-161
lines changed

10 files changed

+354
-161
lines changed
 

‎add-on/_locales/en/messages.json

+35-19
Original file line numberDiff line numberDiff line change
@@ -127,17 +127,13 @@
127127
"message": "This Page",
128128
"description": "An item in right-click context menu (contextMenu_parentPage)"
129129
},
130-
"contextMenu_AddToIpfsKeepFilename": {
131-
"message": "Add to IPFS (Keep Filename)",
132-
"description": "An item in right-click context menu (contextMenu_AddToIpfsKeepFilename)"
130+
"contextMenu_importToIpfs": {
131+
"message": "Import to IPFS",
132+
"description": "An item in right-click context menu (contextMenu_importToIpfs)"
133133
},
134-
"contextMenu_AddToIpfsRawCid": {
135-
"message": "Add to IPFS",
136-
"description": "An item in right-click context menu (contextMenu_AddToIpfsRawCid)"
137-
},
138-
"contextMenu_AddToIpfsSelection": {
139-
"message": "Add Selected Text to IPFS",
140-
"description": "An item in right-click context menu (contextMenu_AddToIpfsSelection)"
134+
"contextMenu_importToIpfsSelection": {
135+
"message": "Import Selected Text to IPFS",
136+
"description": "An item in right-click context menu (contextMenu_importToIpfsSelection)"
141137
},
142138
"notify_addonIssueTitle": {
143139
"message": "IPFS Add-on Issue",
@@ -215,6 +211,10 @@
215211
"message": "IPFS Node",
216212
"description": "A section header on the Preferences screen (option_header_nodeType)"
217213
},
214+
"option_header_fileImport": {
215+
"message": "File Import",
216+
"description": "A section header on the Preferences screen (option_header_fileImport)"
217+
},
218218
"option_ipfsNodeType_title": {
219219
"message": "IPFS Node Type",
220220
"description": "An option title on the Preferences screen (option_ipfsNodeType_title)"
@@ -411,12 +411,20 @@
411411
"message": "Manage permissions",
412412
"description": "Link text for managing permissions"
413413
},
414+
"option_openViaWebUI_title": {
415+
"message": "Open imported files in Web UI",
416+
"description": "An option title on the Preferences screen (option_openViaWebUI_title)"
417+
},
418+
"option_openViaWebUI_description": {
419+
"message": "Display files in Web UI rather than opening file or parent directory at gateway.",
420+
"description": "An option description on the Preferences screen (option_openViaWebUI_description)"
421+
},
414422
"option_preloadAtPublicGateway_title": {
415-
"message": "Preload Uploads",
423+
"message": "Preload Imports",
416424
"description": "An option title on the Preferences screen (option_preloadAtPublicGateway_title)"
417425
},
418426
"option_preloadAtPublicGateway_description": {
419-
"message": "Enables automatic preload of uploaded assets via asynchronous HTTP HEAD request to a Public Gateway",
427+
"message": "Enables automatic preload of imported assets via asynchronous HTTP HEAD request to a Public Gateway",
420428
"description": "An option description on the Preferences screen (option_preloadAtPublicGateway_description)"
421429
},
422430
"option_logNamespaces_title": {
@@ -427,6 +435,14 @@
427435
"message": "Customize which namespaces are logged to Browser Console. Changing this value will trigger extension restart.",
428436
"description": "An option description for the log level (option_logNamespaces_description)"
429437
},
438+
"option_importDir_title": {
439+
"message": "File Import Directory",
440+
"description": "An option title on the Preferences screen (option_importDir_title)"
441+
},
442+
"option_importDir_description": {
443+
"message": "Customize the directory used for imported files.",
444+
"description": "An option description on the Preferences screen (option_importDir_description)"
445+
},
430446
"option_resetAllOptions_title": {
431447
"message": "Reset Everything",
432448
"description": "An option title and button label on the Preferences screen (option_resetAllOptions_title)"
@@ -472,16 +488,16 @@
472488
"description": "Status label on the share files page (quickUpload_state_buffering)"
473489
},
474490
"quickUpload_options_show": {
475-
"message": "upload options",
491+
"message": "import options",
476492
"description": "Button on the share files page (quickUpload_options_show)"
477493
},
478-
"quickUpload_options_wrapWithDirectory": {
479-
"message": "Wrap single files in a directory to preserve their filenames.",
480-
"description": "Checkbox label on the share files page (quickUpload_options_wrapWithDirectory)"
494+
"quickUpload_options_importDir": {
495+
"message": "Path to store imported files",
496+
"description": "Textbox label on the share files page (quickUpload_options_importDir)"
481497
},
482-
"quickUpload_options_pinUpload": {
483-
"message": "Pin files so they are retained when performing garbage collection on your IPFS repo.",
484-
"description": "Checkbox label on the share files page (quickUpload_options_pinUpload)"
498+
"quickUpload_options_openViaWebUI": {
499+
"message": "Open in Web UI",
500+
"description": "Checkbox label on the share files page (quickUpload_options_openViaWebUI)"
485501
},
486502
"page_proxyAcl_title": {
487503
"message": "Manage Permissions",

‎add-on/src/lib/context-menus.js

+6-17
Original file line numberDiff line numberDiff line change
@@ -51,10 +51,9 @@ const menuParentLink = 'contextMenu_parentLink'
5151
const menuParentPage = 'contextMenu_parentPage'
5252
// const menuParentText = 'contextMenu_parentText'
5353
// Generic Add to IPFS
54-
const contextMenuAddToIpfsRawCid = 'contextMenu_AddToIpfsRawCid'
55-
const contextMenuAddToIpfsKeepFilename = 'contextMenu_AddToIpfsKeepFilename'
54+
const contextMenuImportToIpfs = 'contextMenu_importToIpfs'
5655
// Add X to IPFS
57-
const contextMenuAddToIpfsSelection = 'contextMenu_AddToIpfsSelection'
56+
const contextMenuImportToIpfsSelection = 'contextMenu_importToIpfsSelection'
5857
// Copy X
5958
const contextMenuCopyCanonicalAddress = 'panelCopy_currentIpfsAddress'
6059
const contextMenuCopyRawCid = 'panelCopy_copyRawCid'
@@ -78,15 +77,7 @@ function createContextMenus (getState, runtime, ipfsPathValidator, { onAddFromCo
7877
contexts: [contextType]
7978
})
8079
}
81-
const createSeparator = (parentId, id, contextType) => {
82-
return browser.contextMenus.create({
83-
id: `${parentId}_${id}`,
84-
parentId,
85-
type: 'separator',
86-
contexts: ['all']
87-
})
88-
}
89-
const createAddToIpfsMenuItem = (parentId, id, contextType, ipfsAddOptions) => {
80+
const createImportToIpfsMenuItem = (parentId, id, contextType, ipfsAddOptions) => {
9081
const itemId = `${parentId}_${id}`
9182
apiMenuItems.add(itemId)
9283
return browser.contextMenus.create({
@@ -125,19 +116,17 @@ function createContextMenus (getState, runtime, ipfsPathValidator, { onAddFromCo
125116
}
126117
const buildSubmenu = (parentId, contextType) => {
127118
createSubmenu(parentId, contextType)
128-
createAddToIpfsMenuItem(parentId, contextMenuAddToIpfsKeepFilename, contextType, { wrapWithDirectory: true })
129-
createAddToIpfsMenuItem(parentId, contextMenuAddToIpfsRawCid, contextType, { wrapWithDirectory: false })
130-
createSeparator(parentId, 'separator-1', contextType)
119+
createImportToIpfsMenuItem(parentId, contextMenuImportToIpfs, contextType, { wrapWithDirectory: true, pin: false })
131120
createCopierMenuItem(parentId, contextMenuCopyAddressAtPublicGw, contextType, onCopyAddressAtPublicGw)
132121
createCopierMenuItem(parentId, contextMenuCopyCanonicalAddress, contextType, onCopyCanonicalAddress)
133122
createCopierMenuItem(parentId, contextMenuCopyRawCid, contextType, onCopyRawCid)
134123
}
135124

136125
/*
137126
createSubmenu(menuParentText, 'selection')
138-
createAddToIpfsMenuItem(menuParentText, contextMenuAddToIpfsSelection, 'selection')
127+
createImportToIpfsMenuItem(menuParentText, contextMenuImportToIpfsSelection, 'selection')
139128
*/
140-
createAddToIpfsMenuItem(null, contextMenuAddToIpfsSelection, 'selection')
129+
createImportToIpfsMenuItem(null, contextMenuImportToIpfsSelection, 'selection', { pin: false })
141130
buildSubmenu(menuParentImage, 'image')
142131
buildSubmenu(menuParentVideo, 'video')
143132
buildSubmenu(menuParentAudio, 'audio')

‎add-on/src/lib/ipfs-companion.js

+20-59
Original file line numberDiff line numberDiff line change
@@ -12,9 +12,10 @@ const { optionDefaults, storeMissingOptions, migrateOptions } = require('./optio
1212
const { initState, offlinePeerCount } = require('./state')
1313
const { createIpfsPathValidator } = require('./ipfs-path')
1414
const createDnslinkResolver = require('./dnslink')
15-
const { createRequestModifier, redirectOptOutHint } = require('./ipfs-request')
15+
const { createRequestModifier } = require('./ipfs-request')
1616
const { initIpfsClient, destroyIpfsClient } = require('./ipfs-client')
1717
const { createIpfsUrlProtocolHandler } = require('./ipfs-protocol')
18+
const createIpfsImportHandler = require('./ipfs-import')
1819
const createNotifier = require('./notifier')
1920
const createCopier = require('./copier')
2021
const { createRuntimeChecks } = require('./runtime-checks')
@@ -38,6 +39,7 @@ module.exports = async function init () {
3839
var apiStatusUpdateInterval
3940
var ipfsProxy
4041
var ipfsProxyContentScript
42+
var ipfsImportHandler
4143
const idleInSecs = 5 * 60
4244
const browserActionPortName = 'browser-action-port'
4345

@@ -65,6 +67,7 @@ module.exports = async function init () {
6567

6668
dnslinkResolver = createDnslinkResolver(getState)
6769
ipfsPathValidator = createIpfsPathValidator(getState, getIpfs, dnslinkResolver)
70+
ipfsImportHandler = createIpfsImportHandler(getState, getIpfs, ipfsPathValidator, runtime)
6871
copier = createCopier(notify, ipfsPathValidator)
6972
contextMenus = createContextMenus(getState, runtime, ipfsPathValidator, {
7073
onAddFromContext,
@@ -230,6 +233,8 @@ module.exports = async function init () {
230233
gwURLString: dropSlash(state.gwURLString),
231234
pubGwURLString: dropSlash(state.pubGwURLString),
232235
webuiRootUrl: state.webuiRootUrl,
236+
importDir: state.importDir,
237+
openViaWebUI: state.openViaWebUI,
233238
apiURLString: dropSlash(state.apiURLString),
234239
redirect: state.redirect,
235240
noRedirectHostnames: state.noRedirectHostnames,
@@ -257,40 +262,16 @@ module.exports = async function init () {
257262
}
258263
}
259264

260-
// GUI
261-
// ===================================================================
262-
263-
function preloadAtPublicGateway (path) {
264-
if (!state.preloadAtPublicGateway) return
265-
// asynchronous HTTP HEAD request preloads triggers content without downloading it
266-
return new Promise((resolve, reject) => {
267-
const http = new XMLHttpRequest()
268-
// Make sure preload request is excluded from global redirect
269-
const preloadUrl = ipfsPathValidator.resolveToPublicUrl(`${path}#${redirectOptOutHint}`, state.pubGwURLString)
270-
http.open('HEAD', preloadUrl)
271-
http.onreadystatechange = function () {
272-
if (this.readyState === this.DONE) {
273-
console.info(`[ipfs-companion] preloadAtPublicGateway(${path}):`, this.statusText)
274-
if (this.status === 200) {
275-
resolve(this.statusText)
276-
} else {
277-
reject(new Error(this.statusText))
278-
}
279-
}
280-
}
281-
http.send()
282-
})
283-
}
284-
285265
// Context Menu Uploader
286266
// -------------------------------------------------------------------
287267

288268
async function onAddFromContext (context, contextType, options) {
269+
const importDir = ipfsImportHandler.formatImportDirectory(state.importDir)
289270
let result
290271
try {
291272
const dataSrc = await findValueForContext(context, contextType)
292273
if (contextType === 'selection') {
293-
result = await ipfs.add(Buffer.from(dataSrc), options)
274+
result = await ipfsImportHandler.importFiles(Buffer.from(dataSrc), options, importDir)
294275
} else {
295276
// Enchanced addFromURL
296277
// --------------------
@@ -319,7 +300,7 @@ module.exports = async function init () {
319300
path: decodeURIComponent(filename),
320301
content: buffer
321302
}
322-
result = await ipfs.add(data, options)
303+
result = await ipfsImportHandler.importFiles(data, options, importDir)
323304
}
324305
} catch (error) {
325306
console.error('Error in upload to IPFS context menu', error)
@@ -334,37 +315,12 @@ module.exports = async function init () {
334315
}
335316
return
336317
}
337-
338-
return uploadResultHandler({ result, openRootInNewTab: true })
339-
}
340-
341-
// TODO: feature detect and push to client type specific modules.
342-
function getIpfsPathAndNativeAddress (hash) {
343-
const path = `/ipfs/${hash}`
344-
if (runtime.hasNativeProtocolHandler) {
345-
return { path, url: `ipfs://${hash}` }
318+
ipfsImportHandler.preloadFilesAtPublicGateway(result)
319+
if (state.ipfsNodeType === 'embedded' || !state.openViaWebUI) {
320+
return ipfsImportHandler.openFilesAtGateway({ result, openRootInNewTab: true })
346321
} else {
347-
// open at public GW (will be redirected to local elsewhere, if enabled)
348-
const url = new URL(path, state.pubGwURLString).toString()
349-
return { path, url: url }
350-
}
351-
}
352-
353-
async function uploadResultHandler ({ result, openRootInNewTab = false }) {
354-
for (const file of result) {
355-
if (file && file.hash) {
356-
const { path, url } = getIpfsPathAndNativeAddress(file.hash)
357-
preloadAtPublicGateway(path)
358-
console.info('[ipfs-companion] successfully stored', file)
359-
// open the wrapping directory (or the CID if wrapping was disabled)
360-
if (openRootInNewTab && (result.length === 1 || file.path === '' || file.path === file.hash)) {
361-
await browser.tabs.create({
362-
url: url
363-
})
364-
}
365-
}
322+
return ipfsImportHandler.openFilesAtWebUI(importDir)
366323
}
367-
return result
368324
}
369325

370326
// Page-specific Actions
@@ -709,12 +665,16 @@ module.exports = async function init () {
709665
shouldReloadExtension = true
710666
state[key] = localStorage.debug = change.newValue
711667
break
668+
case 'importDir':
669+
state[key] = change.newValue
670+
break
712671
case 'linkify':
713672
case 'catchUnhandledProtocols':
714673
case 'displayNotifications':
715674
case 'automaticMode':
716675
case 'detectIpfsPathHeader':
717676
case 'preloadAtPublicGateway':
677+
case 'openViaWebUI':
718678
case 'noRedirectHostnames':
719679
state[key] = change.newValue
720680
break
@@ -783,8 +743,8 @@ module.exports = async function init () {
783743
return notify
784744
},
785745

786-
get uploadResultHandler () {
787-
return uploadResultHandler
746+
get ipfsImportHandler () {
747+
return ipfsImportHandler
788748
},
789749

790750
destroy () {
@@ -796,6 +756,7 @@ module.exports = async function init () {
796756
dnslinkResolver = null
797757
modifyRequest = null
798758
ipfsPathValidator = null
759+
ipfsImportHandler = null
799760
notify = null
800761
copier = null
801762
contextMenus = null

0 commit comments

Comments
 (0)
Please sign in to comment.