diff --git a/Makefile b/Makefile
index 301b3c3e..9f5b5cdd 100644
--- a/Makefile
+++ b/Makefile
@@ -15,20 +15,32 @@ help:
@echo " "
@echo " Next commands are only for dev environment with nextcloud-docker-dev!"
@echo " Daemon register(Linux, socket):"
- @echo " dock-sock create docker daemon for Nextcloud 30, 29, 28 (/var/run/docker.sock)"
+ @echo " dock-sock create docker daemon for Nextcloud 29, 28, 27 (/var/run/docker.sock)"
+ @echo " dock-sock27 create docker daemon for Nextcloud 27 (/var/run/docker.sock)"
+ @echo " dock-sock27-gpu create docker daemon with GPU for Nextcloud 27 (/var/run/docker.sock)"
@echo " dock-sock28 create docker daemon for Nextcloud 28 (/var/run/docker.sock)"
@echo " dock-sock28-gpu create docker daemon with GPU for Nextcloud 28 (/var/run/docker.sock)"
- @echo " dock-sock29 create docker daemon for Nextcloud 29 (/var/run/docker.sock)"
- @echo " dock-sock29-gpu create docker daemon with GPU for Nextcloud 29 (/var/run/docker.sock)"
@echo " dock-sock create docker daemon for Nextcloud Last (/var/run/docker.sock)"
@echo " dock-sock-gpu create docker daemon with GPU for Nextcloud Last (/var/run/docker.sock)"
@echo " "
@echo " Daemon register(any OS, host:port)"
@echo " dock2port will map docker socket to port. first use this!"
+ @echo " dock-port27 create docker daemon for Nextcloud 27 (host.docker.internal:8443)"
@echo " dock-port28 create docker daemon for Nextcloud 28 (host.docker.internal:8443)"
- @echo " dock-port29 create docker daemon for Nextcloud 29 (host.docker.internal:8443)"
@echo " dock-port create docker daemons for Nextcloud Last (host.docker.internal:8443)"
+.PHONY: dock-sock27
+dock-sock27:
+ @echo "creating daemon for nextcloud 'stable27' container"
+ docker exec master-stable27-1 sudo -u www-data php occ app_api:daemon:register \
+ docker_dev Docker docker-install http /var/run/docker.sock http://stable27.local/index.php --net=master_default
+
+.PHONY: dock-sock27-gpu
+dock-sock27-gpu:
+ @echo "creating daemon with NVIDIA gpu for nextcloud 'stable27' container"
+ docker exec master-stable27-1 sudo -u www-data php occ app_api:daemon:register \
+ docker_dev_gpu "Docker with GPU" docker-install http /var/run/docker.sock http://stable27.local/index.php --net=master_default --gpu --set-default
+
.PHONY: dock-sock28
dock-sock28:
@echo "creating daemon for nextcloud 'stable28' container"
@@ -41,18 +53,6 @@ dock-sock28-gpu:
docker exec master-stable28-1 sudo -u www-data php occ app_api:daemon:register \
docker_dev_gpu "Docker with GPU" docker-install http /var/run/docker.sock http://stable28.local/index.php --net=master_default --gpu --set-default
-.PHONY: dock-sock29
-dock-sock29:
- @echo "creating daemon for nextcloud 'stable29' container"
- docker exec master-stable29-1 sudo -u www-data php occ app_api:daemon:register \
- docker_dev Docker docker-install http /var/run/docker.sock http://stable29.local/index.php --net=master_default
-
-.PHONY: dock-sock29-gpu
-dock-sock29-gpu:
- @echo "creating daemon with NVIDIA gpu for nextcloud 'stable29' container"
- docker exec master-stable29-1 sudo -u www-data php occ app_api:daemon:register \
- docker_dev_gpu "Docker with GPU" docker-install http /var/run/docker.sock http://stable29.local/index.php --net=master_default --gpu --set-default
-
.PHONY: dock-sock
dock-sock:
@echo "creating daemon for nextcloud 'master' container"
@@ -74,20 +74,20 @@ dock2port:
--net=master_default \
--restart unless-stopped --privileged -d ghcr.io/cloud-py-api/nextcloud-appapi-dsp:latest
+.PHONY: dock-port27
+dock-port27:
+ @echo "creating daemon for nextcloud '27' container"
+ docker exec master-stable27-1 sudo -u www-data php occ app_api:daemon:register \
+ docker_dev Docker docker-install http nextcloud-appapi-dsp:2375 http://stable27.local/index.php \
+ --net=master_default --haproxy_password="some_secure_password"
+
.PHONY: dock-port28
dock-port28:
- @echo "creating daemon for nextcloud '28' container"
+ @echo "creating daemon for nextcloud '27' container"
docker exec master-stable28-1 sudo -u www-data php occ app_api:daemon:register \
docker_dev Docker docker-install http nextcloud-appapi-dsp:2375 http://stable28.local/index.php \
--net=master_default --haproxy_password="some_secure_password"
-.PHONY: dock-port29
-dock-port29:
- @echo "creating daemon for nextcloud '29' container"
- docker exec master-stable29-1 sudo -u www-data php occ app_api:daemon:register \
- docker_dev Docker docker-install http nextcloud-appapi-dsp:2375 http://stable29.local/index.php \
- --net=master_default --haproxy_password="some_secure_password"
-
.PHONY: dock-port
dock-port:
@echo "creating daemon for nextcloud 'master' container"
diff --git a/appinfo/info.xml b/appinfo/info.xml
index 78226de2..a4426120 100644
--- a/appinfo/info.xml
+++ b/appinfo/info.xml
@@ -63,7 +63,7 @@ to join us in shaping a more versatile, stable, and secure app landscape.
https://raw.githubusercontent.com/cloud-py-api/app_api/main/screenshots/app_api_4.png
-
+
OCA\AppAPI\BackgroundJob\ExAppInitStatusCheckJob
diff --git a/lib/Listener/LoadFilesPluginListener.php b/lib/Listener/LoadFilesPluginListener.php
index 2969c939..43bcafe0 100644
--- a/lib/Listener/LoadFilesPluginListener.php
+++ b/lib/Listener/LoadFilesPluginListener.php
@@ -36,7 +36,12 @@ public function handle(Event $event): void {
'fileActions' => $exFilesActions,
'instanceId' => $this->config->getSystemValue('instanceid'),
]);
- Util::addScript(Application::APP_ID, Application::APP_ID . '-filesplugin');
+ $ncVersion = $this->config->getSystemValueString('version', '0.0.0');
+ if (version_compare($ncVersion, '28.0', '<')) {
+ Util::addScript(Application::APP_ID, Application::APP_ID . '-filesplugin');
+ } else {
+ Util::addScript(Application::APP_ID, Application::APP_ID . '-filesplugin28');
+ }
Util::addStyle(Application::APP_ID, 'filesactions');
}
}
diff --git a/src/filesplugin.js b/src/filesplugin.js
index 1f606322..97e5707c 100644
--- a/src/filesplugin.js
+++ b/src/filesplugin.js
@@ -1,159 +1,67 @@
import axios from '@nextcloud/axios'
import { generateUrl } from '@nextcloud/router'
import { loadState } from '@nextcloud/initial-state'
-import { registerFileAction, FileAction } from '@nextcloud/files'
-import { getCurrentUser } from '@nextcloud/auth'
import { translate as t } from '@nextcloud/l10n'
+import { getCurrentUser } from '@nextcloud/auth'
const state = loadState('app_api', 'ex_files_actions_menu')
-function loadStaticAppAPIInlineSvgIcon() {
- return ''
-}
-
-function loadExAppInlineSvgIcon(appId, route) {
- const url = generateAppAPIProxyUrl(appId, route)
- return axios.get(url).then((response) => {
- // Check content type to be svg image
- if (response.headers['content-type'] !== 'image/svg+xml') {
- return null
- }
- return response.data
- }).catch((error) => {
- console.error('Failed to load ExApp FileAction icon inline svg', error)
- return null
- })
-}
-
function generateAppAPIProxyUrl(appId, route) {
return generateUrl(`/apps/app_api/proxy/${appId}/${route}`)
}
-function generateExAppUIPageUrl(appId, route) {
- return generateUrl(`/apps/app_api/embedded/${appId}/${route}`)
-}
+document.addEventListener('DOMContentLoaded', () => {
+ if (OCA.Files && OCA.Files.fileActions) { // NC 27
+ state.fileActions.forEach(fileAction => {
+ const mimes = fileAction.mime.split(',').map(mime => mime.trim()) // multiple mimes are separated by comma
-function registerFileAction28(fileAction, inlineSvgIcon) {
- const action = new FileAction({
- id: fileAction.name,
- displayName: () => t(fileAction.appid, fileAction.display_name),
- iconSvgInline: () => inlineSvgIcon,
- order: Number(fileAction.order),
- enabled(files, view) {
- if (files.length === 1) {
- // Check for multiple mimes separated by comma
- let isMimeMatch = false
- fileAction.mime.split(',').forEach((mime) => {
- if (files[0].mime.indexOf(mime.trim()) !== -1) {
- isMimeMatch = true
+ const actionHandler = (fileName, context) => {
+ const file = context.$file[0]
+ const exAppFileActionHandler = generateAppAPIProxyUrl(fileAction.appid, fileAction.action_handler)
+ axios.post(exAppFileActionHandler, {
+ fileId: Number(file.dataset.id),
+ name: fileName,
+ directory: file.dataset.path,
+ etag: file.dataset.etag,
+ mime: file.dataset.mime,
+ favorite: file.dataset.favorite || 'false',
+ permissions: Number(file.dataset.permissions),
+ fileType: file.dataset.type,
+ size: Number(file.dataset.size),
+ mtime: Number(file.dataset.mtime) / 1000, // convert ms to s
+ shareTypes: file.dataset?.shareTypes || null,
+ shareAttributes: file.dataset?.shareAttributes || null,
+ sharePermissions: file.dataset?.sharePermissions || null,
+ shareOwner: file.dataset?.shareOwner || null,
+ shareOwnerId: file.dataset?.shareOwnerId || null,
+ userId: getCurrentUser().uid,
+ instanceId: state.instanceId,
+ }).then((response) => {
+ if (response.status === 200) {
+ OC.dialogs.info(t('app_api', 'Action request sent to ExApp'), t(fileAction.appid, fileAction.display_name))
+ } else {
+ console.debug(response)
+ OC.dialogs.info(t('app_api', 'Error while sending File action request to ExApp'), t(fileAction.appid, fileAction.display_name))
}
+ }).catch((error) => {
+ console.error('error', error)
+ OC.dialogs.info(t('app_api', 'Error while sending File action request to ExApp'), t(fileAction.appid, fileAction.display_name))
})
- return isMimeMatch
- } else if (files.length > 1) {
- // Check all files match fileAction mime
- return files.every((file) => {
- // Check for multiple mimes separated by comma
- let isMimeMatch = false
- fileAction.mime.split(',').forEach((mime) => {
- if (file.mime.indexOf(mime.trim()) !== -1) {
- isMimeMatch = true
- }
- })
- return isMimeMatch
- })
- }
- },
- async exec(node, view, dir) {
- const exAppFileActionHandler = generateAppAPIProxyUrl(fileAction.appid, fileAction.action_handler)
- if ('version' in fileAction && fileAction.version === '2.0') {
- return axios.post(exAppFileActionHandler, { files: [buildNodeInfo(node)] })
- .then((response) => {
- if (typeof response.data === 'object' && 'redirect_handler' in response.data) {
- const redirectPage = generateExAppUIPageUrl(fileAction.appid, response.data.redirect_handler)
- window.location.assign(`${redirectPage}?fileIds=${node.fileid}`)
- return true
- }
- return true
- }).catch((error) => {
- console.error('Failed to send FileAction request to ExApp', error)
- return false
- })
}
- return axios.post(exAppFileActionHandler, buildNodeInfo(node))
- .then(() => {
- return true
- })
- .catch((error) => {
- console.error('Failed to send FileAction request to ExApp', error)
- return false
- })
- },
- async execBatch(nodes, view, dir) {
- if ('version' in fileAction && fileAction.version === '2.0') {
- const exAppFileActionHandler = generateAppAPIProxyUrl(fileAction.appid, fileAction.action_handler)
- const nodesDataList = nodes.map(buildNodeInfo)
- return axios.post(exAppFileActionHandler, { files: nodesDataList })
- .then((response) => {
- if (typeof response.data === 'object' && 'redirect_handler' in response.data) {
- const redirectPage = generateExAppUIPageUrl(fileAction.appid, response.data.redirect_handler)
- const fileIds = nodes.map((node) => node.fileid).join(',')
- window.location.assign(`${redirectPage}?fileIds=${fileIds}`)
- }
- return nodes.map(_ => true)
- })
- .catch((error) => {
- console.error('Failed to send FileAction request to ExApp', error)
- return nodes.map(_ => false)
- })
- }
- // for version 1.0 behavior is not changed
- return Promise.all(nodes.map((node) => {
- return this.exec(node, view, dir)
- }))
- },
- })
- registerFileAction(action)
-}
-function buildNodeInfo(node) {
- return {
- fileId: node.fileid,
- name: node.basename,
- directory: node.dirname,
- etag: node.attributes.etag,
- mime: node.mime,
- favorite: Boolean(node.attributes.favorite).toString(),
- permissions: node.permissions,
- fileType: node.type,
- size: Number(node.size),
- mtime: new Date(node.mtime).getTime() / 1000, // convert ms to s
- shareTypes: node.attributes.shareTypes || null,
- shareAttributes: node.attributes.shareAttributes || null,
- sharePermissions: node.attributes.sharePermissions || null,
- shareOwner: node.attributes.ownerDisplayName || null,
- shareOwnerId: node.attributes.ownerId || null,
- userId: getCurrentUser().uid,
- instanceId: state.instanceId,
- }
-}
-
-document.addEventListener('DOMContentLoaded', () => {
- state.fileActions.forEach(fileAction => {
- if (fileAction.icon === '') {
- const inlineSvgIcon = loadStaticAppAPIInlineSvgIcon()
- registerFileAction28(fileAction, inlineSvgIcon)
- } else {
- loadExAppInlineSvgIcon(fileAction.appid, fileAction.icon).then((svg) => {
- if (svg !== null) {
- // Set css filter for theming
- const parser = new DOMParser()
- const icon = parser.parseFromString(svg, 'image/svg+xml')
- icon.documentElement.setAttribute('style', 'filter: var(--background-invert-if-dark);')
- // Convert back to inline string
- const inlineSvgIcon = icon.documentElement.outerHTML
- registerFileAction28(fileAction, inlineSvgIcon)
+ mimes.forEach((mimeType) => {
+ const action = {
+ name: fileAction.name,
+ displayName: t(fileAction.appid, fileAction.display_name),
+ mime: mimeType,
+ permissions: Number(fileAction.permissions),
+ order: Number(fileAction.order),
+ icon: fileAction.icon !== '' ? generateAppAPIProxyUrl(fileAction.appid, fileAction.icon) : null,
+ iconClass: fileAction.icon === '' ? 'icon-app-api' : '',
+ actionHandler,
}
+ OCA.Files.fileActions.registerAction(action)
})
- }
- })
+ })
+ }
})
diff --git a/src/filesplugin28.js b/src/filesplugin28.js
new file mode 100644
index 00000000..1f606322
--- /dev/null
+++ b/src/filesplugin28.js
@@ -0,0 +1,159 @@
+import axios from '@nextcloud/axios'
+import { generateUrl } from '@nextcloud/router'
+import { loadState } from '@nextcloud/initial-state'
+import { registerFileAction, FileAction } from '@nextcloud/files'
+import { getCurrentUser } from '@nextcloud/auth'
+import { translate as t } from '@nextcloud/l10n'
+
+const state = loadState('app_api', 'ex_files_actions_menu')
+
+function loadStaticAppAPIInlineSvgIcon() {
+ return ''
+}
+
+function loadExAppInlineSvgIcon(appId, route) {
+ const url = generateAppAPIProxyUrl(appId, route)
+ return axios.get(url).then((response) => {
+ // Check content type to be svg image
+ if (response.headers['content-type'] !== 'image/svg+xml') {
+ return null
+ }
+ return response.data
+ }).catch((error) => {
+ console.error('Failed to load ExApp FileAction icon inline svg', error)
+ return null
+ })
+}
+
+function generateAppAPIProxyUrl(appId, route) {
+ return generateUrl(`/apps/app_api/proxy/${appId}/${route}`)
+}
+
+function generateExAppUIPageUrl(appId, route) {
+ return generateUrl(`/apps/app_api/embedded/${appId}/${route}`)
+}
+
+function registerFileAction28(fileAction, inlineSvgIcon) {
+ const action = new FileAction({
+ id: fileAction.name,
+ displayName: () => t(fileAction.appid, fileAction.display_name),
+ iconSvgInline: () => inlineSvgIcon,
+ order: Number(fileAction.order),
+ enabled(files, view) {
+ if (files.length === 1) {
+ // Check for multiple mimes separated by comma
+ let isMimeMatch = false
+ fileAction.mime.split(',').forEach((mime) => {
+ if (files[0].mime.indexOf(mime.trim()) !== -1) {
+ isMimeMatch = true
+ }
+ })
+ return isMimeMatch
+ } else if (files.length > 1) {
+ // Check all files match fileAction mime
+ return files.every((file) => {
+ // Check for multiple mimes separated by comma
+ let isMimeMatch = false
+ fileAction.mime.split(',').forEach((mime) => {
+ if (file.mime.indexOf(mime.trim()) !== -1) {
+ isMimeMatch = true
+ }
+ })
+ return isMimeMatch
+ })
+ }
+ },
+ async exec(node, view, dir) {
+ const exAppFileActionHandler = generateAppAPIProxyUrl(fileAction.appid, fileAction.action_handler)
+ if ('version' in fileAction && fileAction.version === '2.0') {
+ return axios.post(exAppFileActionHandler, { files: [buildNodeInfo(node)] })
+ .then((response) => {
+ if (typeof response.data === 'object' && 'redirect_handler' in response.data) {
+ const redirectPage = generateExAppUIPageUrl(fileAction.appid, response.data.redirect_handler)
+ window.location.assign(`${redirectPage}?fileIds=${node.fileid}`)
+ return true
+ }
+ return true
+ }).catch((error) => {
+ console.error('Failed to send FileAction request to ExApp', error)
+ return false
+ })
+ }
+ return axios.post(exAppFileActionHandler, buildNodeInfo(node))
+ .then(() => {
+ return true
+ })
+ .catch((error) => {
+ console.error('Failed to send FileAction request to ExApp', error)
+ return false
+ })
+ },
+ async execBatch(nodes, view, dir) {
+ if ('version' in fileAction && fileAction.version === '2.0') {
+ const exAppFileActionHandler = generateAppAPIProxyUrl(fileAction.appid, fileAction.action_handler)
+ const nodesDataList = nodes.map(buildNodeInfo)
+ return axios.post(exAppFileActionHandler, { files: nodesDataList })
+ .then((response) => {
+ if (typeof response.data === 'object' && 'redirect_handler' in response.data) {
+ const redirectPage = generateExAppUIPageUrl(fileAction.appid, response.data.redirect_handler)
+ const fileIds = nodes.map((node) => node.fileid).join(',')
+ window.location.assign(`${redirectPage}?fileIds=${fileIds}`)
+ }
+ return nodes.map(_ => true)
+ })
+ .catch((error) => {
+ console.error('Failed to send FileAction request to ExApp', error)
+ return nodes.map(_ => false)
+ })
+ }
+ // for version 1.0 behavior is not changed
+ return Promise.all(nodes.map((node) => {
+ return this.exec(node, view, dir)
+ }))
+ },
+ })
+ registerFileAction(action)
+}
+
+function buildNodeInfo(node) {
+ return {
+ fileId: node.fileid,
+ name: node.basename,
+ directory: node.dirname,
+ etag: node.attributes.etag,
+ mime: node.mime,
+ favorite: Boolean(node.attributes.favorite).toString(),
+ permissions: node.permissions,
+ fileType: node.type,
+ size: Number(node.size),
+ mtime: new Date(node.mtime).getTime() / 1000, // convert ms to s
+ shareTypes: node.attributes.shareTypes || null,
+ shareAttributes: node.attributes.shareAttributes || null,
+ sharePermissions: node.attributes.sharePermissions || null,
+ shareOwner: node.attributes.ownerDisplayName || null,
+ shareOwnerId: node.attributes.ownerId || null,
+ userId: getCurrentUser().uid,
+ instanceId: state.instanceId,
+ }
+}
+
+document.addEventListener('DOMContentLoaded', () => {
+ state.fileActions.forEach(fileAction => {
+ if (fileAction.icon === '') {
+ const inlineSvgIcon = loadStaticAppAPIInlineSvgIcon()
+ registerFileAction28(fileAction, inlineSvgIcon)
+ } else {
+ loadExAppInlineSvgIcon(fileAction.appid, fileAction.icon).then((svg) => {
+ if (svg !== null) {
+ // Set css filter for theming
+ const parser = new DOMParser()
+ const icon = parser.parseFromString(svg, 'image/svg+xml')
+ icon.documentElement.setAttribute('style', 'filter: var(--background-invert-if-dark);')
+ // Convert back to inline string
+ const inlineSvgIcon = icon.documentElement.outerHTML
+ registerFileAction28(fileAction, inlineSvgIcon)
+ }
+ })
+ }
+ })
+})
diff --git a/webpack.js b/webpack.js
index 203ae358..a40f03fc 100644
--- a/webpack.js
+++ b/webpack.js
@@ -22,6 +22,7 @@ webpackConfig.entry = {
main: { import: path.join(__dirname, 'src', 'main.js'), filename: appId + '-main.js' },
adminSettings: { import: path.join(__dirname, 'src', 'adminSettings.js'), filename: appId + '-adminSettings.js' },
filesplugin: { import: path.join(__dirname, 'src', 'filesplugin.js'), filename: appId + '-filesplugin.js' },
+ filesplugin28: { import: path.join(__dirname, 'src', 'filesplugin28.js'), filename: appId + '-filesplugin28.js' },
}
webpackConfig.plugins.push(