Skip to content

Commit

Permalink
feat(sharing): Rework logic of ShareRestrictionModal
Browse files Browse the repository at this point in the history
Cozy-client has been updated, allowing us to make
a few optimizations here.

The `forwardFile` function has no place here,
as it is only used on mobile (which is another component).
  • Loading branch information
Merkur39 committed Jan 17, 2025
1 parent fd145fa commit 3a3740c
Show file tree
Hide file tree
Showing 5 changed files with 232 additions and 86 deletions.
13 changes: 9 additions & 4 deletions packages/cozy-sharing/src/SharingProvider.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -331,23 +331,28 @@ export class SharingProvider extends Component {
* updateDocumentPermissions - Description
*
* @param {Object} document A shared document
* @param {Array} newVerbs The new verbs to use for the permission, eg. ['GET']
* @param {object} options The new verbs to use for the permission, eg. ['GET']
* @param {string[]} options.verbs The new verbs to use for the permission, eg. ['GET']
* @param {string} [options.expiresAt] The new expiration date for the permission
* @param {string} [options.password] The new password for the permission
*
* @return {Array}
*/
updateDocumentPermissions = async (document, newVerbs) => {
updateDocumentPermissions = async (document, options) => {
const { verbs, expiresAt, password } = options
const permissions = getDocumentPermissions(this.state, document.id)

const responses = await Promise.all(
permissions.map(async permissionDocument => {
const updatedPermissions = permissionDocument.attributes.permissions
Object.keys(updatedPermissions).map(permType => {
updatedPermissions[permType].verbs = newVerbs
updatedPermissions[permType].verbs = verbs
})

const resp = await this.permissionCol.add(
permissionDocument,
updatedPermissions
updatedPermissions,
{ expiresAt, password }
)
this.dispatch(updateSharingLink(resp))
return resp
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,7 @@ import { addDays } from 'date-fns'
import PropTypes from 'prop-types'
import React, { useState } from 'react'

import { useClient } from 'cozy-client'
import { getSharingLink as makeSharingLink } from 'cozy-client/dist/models/sharing'
import { isMobile } from 'cozy-device-helper'
import { generateWebLink, useClient } from 'cozy-client'
import Button from 'cozy-ui/transpiled/react/Buttons'
import { ConfirmDialog } from 'cozy-ui/transpiled/react/CozyDialogs'
import Icon from 'cozy-ui/transpiled/react/Icon'
Expand All @@ -14,14 +12,17 @@ import { useI18n } from 'cozy-ui/transpiled/react/providers/I18n'
import { ShareRestrictionContentModal } from './ShareRestrictionContentModal'
import {
copyToClipboard,
forwardFile,
updatePermissions,
makeTTL,
READ_ONLY_PERMS,
WRITE_PERMS,
revokePermissions
revokePermissions,
createPermissions
} from './helpers'
import { checkIsReadOnlyPermissions } from '../../helpers/permissions'
import {
checkIsPermissionHasExpiresDate,
checkIsPermissionHasPassword,
checkIsReadOnlyPermissions,
getPermissionExpiresDate
} from '../../helpers/permissions'
import { useSharingContext } from '../../hooks/useSharingContext'

const PASSWORD_MIN_LENGTH = 4
Expand All @@ -31,11 +32,8 @@ export const ShareRestrictionModal = ({ file, onClose }) => {
const { t } = useI18n()
const { showAlert } = useAlert()
const [password, setPassword] = useState('')
const [selectedDate, setSelectedDate] = useState(addDays(new Date(), 30))
const [isValidDate, setIsValidDate] = useState(true)
const [isValidPassword, setIsValidPassword] = useState(true)
const [dateToggle, setDateToggle] = useState(true)
const [passwordToggle, setPasswordToggle] = useState(false)
const [loading, setLoading] = useState(false)

const {
Expand All @@ -50,14 +48,22 @@ export const ShareRestrictionModal = ({ file, onClose }) => {
const hasSharingLink = getSharingLink(file._id) !== null
const permissions = getDocumentPermissions(file._id)
const isReadOnlyPermissions = checkIsReadOnlyPermissions(permissions)

const hasPassword = checkIsPermissionHasPassword(permissions)
const hasExpiresDate = checkIsPermissionHasExpiresDate(permissions)
const expiresDate = getPermissionExpiresDate(permissions)
const defaultDate = expiresDate
? new Date(expiresDate)
: addDays(new Date(), 30)

const [selectedDate, setSelectedDate] = useState(defaultDate)
const [dateToggle, setDateToggle] = useState(
permissions.length > 0 ? hasExpiresDate : true
)
const [passwordToggle, setPasswordToggle] = useState(hasPassword)
const [editingRights, setEditingRights] = useState(
isReadOnlyPermissions || permissions.length === 0 ? 'readOnly' : 'write'
)

const isDesktopOrMobileWithoutShareAPI =
(isMobile() && !navigator.share) || !isMobile()

const helperTextPassword = !isValidPassword
? t('ShareRestrictionModal.invalidPasswordMessage', {
smart_count: PASSWORD_MIN_LENGTH - password.length
Expand All @@ -73,41 +79,49 @@ export const ShareRestrictionModal = ({ file, onClose }) => {
setLoading(true)
// If the file is not shared, we create a new sharing link
if (!hasSharingLink) {
const verbs = editingRights === 'readOnly' ? READ_ONLY_PERMS : WRITE_PERMS
await shareByLink(file, { verbs })
const url = getSharingLink(file._id)
await copyToClipboard(url, { t, showAlert })
onClose()
return
}

await updatePermissions({
file,
t,
editingRights,
documentType,
updateDocumentPermissions,
showAlert
})

const ttl = makeTTL(dateToggle && selectedDate)
if (isDesktopOrMobileWithoutShareAPI) {
const url = await makeSharingLink(client, [file._id], {
const ttl = makeTTL(dateToggle && selectedDate)
const { data: perms } = await createPermissions({
file,
t,
ttl,
password
password,
editingRights,
documentType,
shareByLink,
showAlert
})
const url = generateWebLink({
cozyUrl: client.getStackClient().uri,
searchParams: [['sharecode', perms.attributes.shortcodes.code]],
pathname: '/public',
slug: 'drive',
subDomainType: client.capabilities.flat_subdomains ? 'flat' : 'nested'
})
await copyToClipboard(url, { t, showAlert })
onClose()
} else {
await forwardFile({
client,
const [{ data: perms }] = await updatePermissions({
file,
t,
ttl,
dateToggle,
selectedDate,
passwordToggle,
password,
editingRights,
documentType,
updateDocumentPermissions,
showAlert
})
const url = generateWebLink({
cozyUrl: client.getStackClient().uri,
searchParams: [['sharecode', perms.attributes.shortcodes.code]],
pathname: '/public',
slug: 'drive',
subDomainType: client.capabilities.flat_subdomains ? 'flat' : 'nested'
})
await copyToClipboard(url, { t, showAlert })
onClose()
}
onClose()
}

const handleRevokeLink = async () => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -109,11 +109,53 @@ export const forwardFile = async ({
}
}

/**
* createPermissions - Create the permissions of a file
* @param {object} options
* @param {import('cozy-client/types/types').IOCozyFile} options.file File to update permissions
* @param {Function} options.t i18n function
* @param {Date|string} options.ttl - Time to live of the sharing link
* @param {string} options.password - Password
* @param {'readOnly'|'write'} options.editingRights - Editing rights
* @param {string} options.documentType - Type of the document
* @param {Function} options.shareByLink - Function to create permissions
* @param {Function} options.showAlert - Function to display an alert
*/
export const createPermissions = async ({
file,
t,
ttl,
password,
editingRights,
documentType,
shareByLink,
showAlert
}) => {
try {
const verbs = editingRights === 'readOnly' ? READ_ONLY_PERMS : WRITE_PERMS
return shareByLink(file, { verbs, ttl, password })
} catch (err) {
showAlert({
message: t(`${documentType}.share.shareByLink.permserror`),
severity: 'error',
variant: 'filled'
})
log.error(
"Error in 'readOnlyPermissionLink' function when trying to change permission",
err
)
}
}

/**
* updatePermissions - Updates the permissions of a file
* @param {object} options
* @param {import('cozy-client/types/types').IOCozyFile} options.file File to update permissions
* @param {Function} options.t i18n function
* @param {boolean} options.dateToggle - Expiration date toggle
* @param {Date|null} options.selectedDate - Expiration date
* @param {boolean} options.passwordToggle - Password toggle
* @param {string} options.password - Password
* @param {string} options.documentType - Type of the document
* @param {'readOnly'|'write'} options.editingRights - Editing rights
* @param {Function} options.updateDocumentPermissions - Function to update permissions
Expand All @@ -122,42 +164,34 @@ export const forwardFile = async ({
export const updatePermissions = async ({
file,
t,
dateToggle,
selectedDate,
passwordToggle,
password,
editingRights,
documentType,
updateDocumentPermissions,
showAlert
}) => {
switch (editingRights) {
case 'readOnly':
try {
return updateDocumentPermissions(file, READ_ONLY_PERMS)
} catch (err) {
showAlert({
message: t(`${documentType}.share.shareByLink.permserror`),
severity: 'error',
variant: 'filled'
})
log.error(
"Error in 'readOnlyPermissionLink' function when trying to change permission",
err
)
}
break
case 'write':
try {
return updateDocumentPermissions(file, WRITE_PERMS)
} catch (err) {
showAlert({
message: t(`${documentType}.share.shareByLink.permserror`),
severity: 'error',
variant: 'filled'
})
log.error(
"Error in 'editPermissionLink' function when trying to change permission",
err
)
}
break
try {
const expiresAt = dateToggle ? selectedDate?.toISOString() || undefined : ''
const ensurePassword = passwordToggle ? password || undefined : ''
const verbs = editingRights === 'readOnly' ? READ_ONLY_PERMS : WRITE_PERMS
return updateDocumentPermissions(file, {
verbs,
expiresAt,
password: ensurePassword
})
} catch (err) {
showAlert({
message: t(`${documentType}.share.shareByLink.permserror`),
severity: 'error',
variant: 'filled'
})
log.error(
"Error in 'updateDocumentPermissions' function when trying to change permission",
err
)
}
}

Expand Down
Loading

0 comments on commit 3a3740c

Please sign in to comment.