mirror of
https://github.com/overleaf/overleaf.git
synced 2025-04-05 22:01:38 +00:00
Merge pull request #3561 from overleaf/ta-file-tree-move-fixes
[ReactFileTree] Move & Rename Validation Fixes GitOrigin-RevId: 30c2ff17454246c82a03d0b869a810048c9f5dad
This commit is contained in:
parent
4d6d4f7f5b
commit
e1ffeef06a
7 changed files with 154 additions and 25 deletions
|
@ -24,6 +24,8 @@
|
|||
"fast",
|
||||
"file_outline",
|
||||
"file_already_exists",
|
||||
"file_already_exists_in_this_location",
|
||||
"blocked_filename",
|
||||
"files_cannot_include_invalid_characters",
|
||||
"find_out_more_about_the_file_outline",
|
||||
"first_error_popup_label",
|
||||
|
|
|
@ -1,18 +1,25 @@
|
|||
import React from 'react'
|
||||
|
||||
import { Button, Modal } from 'react-bootstrap'
|
||||
import { useTranslation } from 'react-i18next'
|
||||
import { Trans, useTranslation } from 'react-i18next'
|
||||
|
||||
import { useFileTreeActionable } from '../../contexts/file-tree-actionable'
|
||||
|
||||
import { InvalidFilenameError, DuplicateFilenameError } from '../../errors'
|
||||
import {
|
||||
InvalidFilenameError,
|
||||
BlockedFilenameError,
|
||||
DuplicateFilenameError,
|
||||
DuplicateFilenameMoveError
|
||||
} from '../../errors'
|
||||
|
||||
function FileTreeModalError() {
|
||||
const { t } = useTranslation()
|
||||
|
||||
const { isRenaming, cancel, error } = useFileTreeActionable()
|
||||
const { isRenaming, isMoving, cancel, error } = useFileTreeActionable()
|
||||
|
||||
if (!isRenaming || !error) return null // the modal will not be rendered; return early
|
||||
// the modal will not be rendered; return early
|
||||
if (!error) return null
|
||||
if (!isRenaming && !isMoving) return null
|
||||
|
||||
function handleHide() {
|
||||
cancel()
|
||||
|
@ -21,8 +28,10 @@ function FileTreeModalError() {
|
|||
function errorTitle() {
|
||||
switch (error.constructor) {
|
||||
case DuplicateFilenameError:
|
||||
case DuplicateFilenameMoveError:
|
||||
return t('duplicate_file')
|
||||
case InvalidFilenameError:
|
||||
case BlockedFilenameError:
|
||||
return t('invalid_file_name')
|
||||
default:
|
||||
return t('error')
|
||||
|
@ -33,8 +42,18 @@ function FileTreeModalError() {
|
|||
switch (error.constructor) {
|
||||
case DuplicateFilenameError:
|
||||
return t('file_already_exists')
|
||||
case DuplicateFilenameMoveError:
|
||||
return (
|
||||
<Trans
|
||||
i18nKey="file_already_exists_in_this_location"
|
||||
components={[<strong />]} // eslint-disable-line react/jsx-key
|
||||
values={{ fileName: error.entityName }}
|
||||
/>
|
||||
)
|
||||
case InvalidFilenameError:
|
||||
return t('files_cannot_include_invalid_characters')
|
||||
case BlockedFilenameError:
|
||||
return t('blocked_filename')
|
||||
default:
|
||||
return t('generic_something_went_wrong')
|
||||
}
|
||||
|
|
|
@ -11,13 +11,18 @@ import {
|
|||
} from '../util/sync-mutation'
|
||||
import { findInTreeOrThrow } from '../util/find-in-tree'
|
||||
import { isNameUniqueInFolder } from '../util/is-name-unique-in-folder'
|
||||
import { isCleanFilename } from '../util/safe-path'
|
||||
import { isBlockedFilename, isCleanFilename } from '../util/safe-path'
|
||||
|
||||
import { FileTreeMainContext } from './file-tree-main'
|
||||
import { useFileTreeMutable } from './file-tree-mutable'
|
||||
import { useFileTreeSelectable } from './file-tree-selectable'
|
||||
|
||||
import { InvalidFilenameError, DuplicateFilenameError } from '../errors'
|
||||
import {
|
||||
InvalidFilenameError,
|
||||
BlockedFilenameError,
|
||||
DuplicateFilenameError,
|
||||
DuplicateFilenameMoveError
|
||||
} from '../errors'
|
||||
|
||||
const FileTreeActionableContext = createContext()
|
||||
|
||||
|
@ -27,6 +32,7 @@ const ACTION_TYPES = {
|
|||
DELETING: 'DELETING',
|
||||
START_CREATE_FOLDER: 'START_CREATE_FOLDER',
|
||||
CREATING_FOLDER: 'CREATING_FOLDER',
|
||||
MOVING: 'MOVING',
|
||||
CANCEL: 'CANCEL',
|
||||
CLEAR: 'CLEAR',
|
||||
ERROR: 'ERROR'
|
||||
|
@ -36,6 +42,7 @@ const defaultState = {
|
|||
isDeleting: false,
|
||||
isRenaming: false,
|
||||
isCreatingFolder: false,
|
||||
isMoving: false,
|
||||
inFlight: false,
|
||||
actionedEntities: null,
|
||||
error: null
|
||||
|
@ -68,6 +75,12 @@ function fileTreeActionableReducer(state, action) {
|
|||
inFlight: true,
|
||||
actionedEntities: state.actionedEntities
|
||||
}
|
||||
case ACTION_TYPES.MOVING:
|
||||
return {
|
||||
...defaultState,
|
||||
isMoving: true,
|
||||
inFlight: true
|
||||
}
|
||||
case ACTION_TYPES.CLEAR:
|
||||
return { ...defaultState }
|
||||
case ACTION_TYPES.CANCEL:
|
||||
|
@ -107,6 +120,7 @@ export function useFileTreeActionable() {
|
|||
const {
|
||||
isDeleting,
|
||||
isRenaming,
|
||||
isMoving,
|
||||
isCreatingFolder,
|
||||
inFlight,
|
||||
error,
|
||||
|
@ -130,18 +144,9 @@ export function useFileTreeActionable() {
|
|||
if (newName === oldName) {
|
||||
return dispatch({ type: ACTION_TYPES.CLEAR })
|
||||
}
|
||||
let error
|
||||
// check valid name
|
||||
if (!isCleanFilename(newName)) {
|
||||
error = new InvalidFilenameError()
|
||||
return dispatch({ type: ACTION_TYPES.ERROR, error })
|
||||
}
|
||||
|
||||
// check for duplicates
|
||||
if (!isNameUniqueInFolder(fileTreeData, found.parentFolderId, newName)) {
|
||||
error = new DuplicateFilenameError()
|
||||
return dispatch({ type: ACTION_TYPES.ERROR, error })
|
||||
}
|
||||
const error = validateRename(fileTreeData, found, newName)
|
||||
if (error) return dispatch({ type: ACTION_TYPES.ERROR, error })
|
||||
|
||||
dispatch({ type: ACTION_TYPES.CLEAR })
|
||||
dispatchRename(selectedEntityId, newName)
|
||||
|
@ -191,14 +196,35 @@ export function useFileTreeActionable() {
|
|||
|
||||
// moves entities. Tree is updated immediately and data are sync'd after.
|
||||
function finishMoving(toFolderId, draggedEntityIds) {
|
||||
draggedEntityIds.forEach(selectedEntityId => {
|
||||
dispatchMove(selectedEntityId, toFolderId)
|
||||
})
|
||||
dispatch({ type: ACTION_TYPES.MOVING })
|
||||
|
||||
return mapSeries(Array.from(draggedEntityIds), id => {
|
||||
const found = findInTreeOrThrow(fileTreeData, id)
|
||||
return syncMove(projectId, found.type, found.entity._id, toFolderId)
|
||||
})
|
||||
// find entities and filter out no-ops
|
||||
const founds = Array.from(draggedEntityIds)
|
||||
.map(draggedEntityId => findInTreeOrThrow(fileTreeData, draggedEntityId))
|
||||
.filter(found => found.parentFolderId !== toFolderId)
|
||||
|
||||
// make sure all entities can be moved, return early otherwise
|
||||
const isMoveToRoot = toFolderId === fileTreeData._id
|
||||
const validationError = founds
|
||||
.map(found => validateMove(fileTreeData, toFolderId, found, isMoveToRoot))
|
||||
.find(error => error)
|
||||
if (validationError) {
|
||||
return dispatch({ type: ACTION_TYPES.ERROR, error: validationError })
|
||||
}
|
||||
|
||||
// dispatch moves immediately
|
||||
founds.forEach(found => dispatchMove(found.entity._id, toFolderId))
|
||||
|
||||
// sync dispatched moves after
|
||||
return mapSeries(founds, found =>
|
||||
syncMove(projectId, found.type, found.entity._id, toFolderId)
|
||||
)
|
||||
.then(() => {
|
||||
dispatch({ type: ACTION_TYPES.CLEAR })
|
||||
})
|
||||
.catch(error => {
|
||||
dispatch({ type: ACTION_TYPES.ERROR, error })
|
||||
})
|
||||
}
|
||||
|
||||
function startCreatingFolder() {
|
||||
|
@ -308,6 +334,7 @@ export function useFileTreeActionable() {
|
|||
canRename: selectedEntityIds.size === 1,
|
||||
canCreate: selectedEntityIds.size < 2,
|
||||
isDeleting,
|
||||
isMoving,
|
||||
isRenaming,
|
||||
isCreatingFolder,
|
||||
inFlight,
|
||||
|
@ -339,3 +366,32 @@ function getSelectedParentFolderId(fileTreeData, selectedEntityIds) {
|
|||
const found = findInTreeOrThrow(fileTreeData, selectedEntityId)
|
||||
return found.type === 'folder' ? found.entity._id : found.parentFolderId
|
||||
}
|
||||
|
||||
function validateRename(fileTreeData, found, newName) {
|
||||
if (!isCleanFilename(newName)) {
|
||||
return new InvalidFilenameError()
|
||||
}
|
||||
|
||||
if (!isNameUniqueInFolder(fileTreeData, found.parentFolderId, newName)) {
|
||||
return new DuplicateFilenameError()
|
||||
}
|
||||
|
||||
const isTopLevel = found.path.length === 1
|
||||
const isFolder = found.type === 'folder'
|
||||
if (isTopLevel && !isFolder && isBlockedFilename(newName)) {
|
||||
return new BlockedFilenameError()
|
||||
}
|
||||
}
|
||||
|
||||
function validateMove(fileTreeData, toFolderId, found, isMoveToRoot) {
|
||||
if (!isNameUniqueInFolder(fileTreeData, toFolderId, found.entity.name)) {
|
||||
const error = new DuplicateFilenameMoveError()
|
||||
error.entityName = found.entity.name
|
||||
return error
|
||||
}
|
||||
|
||||
const isFolder = found.type === 'folder'
|
||||
if (isMoveToRoot && !isFolder && isBlockedFilename(found.entity.name)) {
|
||||
return new BlockedFilenameError()
|
||||
}
|
||||
}
|
||||
|
|
|
@ -4,8 +4,20 @@ export class InvalidFilenameError extends Error {
|
|||
}
|
||||
}
|
||||
|
||||
export class BlockedFilenameError extends Error {
|
||||
constructor() {
|
||||
super('blocked filename')
|
||||
}
|
||||
}
|
||||
|
||||
export class DuplicateFilenameError extends Error {
|
||||
constructor() {
|
||||
super('duplicate filename')
|
||||
}
|
||||
}
|
||||
|
||||
export class DuplicateFilenameMoveError extends Error {
|
||||
constructor() {
|
||||
super('duplicate filename on move')
|
||||
}
|
||||
}
|
||||
|
|
|
@ -76,6 +76,34 @@ export function isCleanFilename(filename) {
|
|||
)
|
||||
}
|
||||
|
||||
export function isBlockedFilename(filename) {
|
||||
return BLOCKEDFILE_RX.test(filename)
|
||||
}
|
||||
|
||||
// returns whether a full path is 'clean' - e.g. is a full or relative path
|
||||
// that points to a file, and each element passes the rules in 'isCleanFilename'
|
||||
export function isCleanPath(path) {
|
||||
const elements = path.split('/')
|
||||
|
||||
const lastElementIsEmpty = elements[elements.length - 1].length === 0
|
||||
if (lastElementIsEmpty) {
|
||||
return false
|
||||
}
|
||||
|
||||
for (let element of Array.from(elements)) {
|
||||
if (element.length > 0 && !isCleanFilename(element)) {
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
// check for a top-level reserved name
|
||||
if (BLOCKEDFILE_RX.test(path.replace(/^\/?/, ''))) {
|
||||
return false
|
||||
} // remove leading slash if present
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
export function isAllowedLength(pathname) {
|
||||
return pathname.length > 0 && pathname.length <= MAX_PATH
|
||||
}
|
||||
|
|
|
@ -93,7 +93,8 @@
|
|||
"confirmation_token_invalid": "Sorry, your confirmation token is invalid or has expired. Please request a new email confirmation link.",
|
||||
"confirmation_link_broken": "Sorry, something is wrong with your confirmation link. Please try copy and pasting the link from the bottom of your confirmation email.",
|
||||
"duplicate_file": "Duplicate File",
|
||||
"file_already_exists_in_this_location": "An item named __fileName__ already exists in this location. If you wish to move this file, rename or remove the conflicting file and try again.",
|
||||
"file_already_exists_in_this_location": "An item named <0>__fileName__</0> already exists in this location. If you wish to move this file, rename or remove the conflicting file and try again.",
|
||||
"blocked_filename": "This file name is blocked.",
|
||||
"group_full": "This group is already full",
|
||||
"no_selection_select_file": "Currently, no file is selected. Please select a file from the file tree.",
|
||||
"no_selection_create_new_file": "Your project is currently empty. Please create a new file.",
|
||||
|
|
|
@ -150,6 +150,17 @@ describe('FileTree Rename Entity Flow', function() {
|
|||
})
|
||||
})
|
||||
|
||||
it('shows error modal on blocked filename', async function() {
|
||||
const input = initItemRename('a.tex')
|
||||
fireEvent.change(input, { target: { value: 'prototype' } })
|
||||
fireEvent.keyDown(input, { key: 'Enter' })
|
||||
|
||||
await screen.findByRole('alert', {
|
||||
name: 'This file name is blocked.',
|
||||
hidden: true
|
||||
})
|
||||
})
|
||||
|
||||
describe('via socket event', function() {
|
||||
it('renames doc', function() {
|
||||
screen.getByRole('treeitem', { name: 'a.tex' })
|
||||
|
|
Loading…
Reference in a new issue