Merge pull request #384 from readur/feat/show-sync-url-preview

feat(ui): show in the UI the sync URL that would be hit
This commit is contained in:
Jon Fuller
2025-12-11 12:56:32 -08:00
committed by GitHub
2 changed files with 860 additions and 28 deletions

View File

@@ -279,6 +279,168 @@ const SourcesPage: React.FC = () => {
}
};
// Helper function to build example sync URL based on source type and configuration
const buildExampleSyncUrl = (): { parts: { text: string; type: 'server' | 'path' | 'folder' | 'file' }[] } | null => {
const exampleFile = 'document1.pdf';
const firstFolder = formData.watch_folders.length > 0 ? formData.watch_folders[0] : '/Documents';
if (formData.source_type === 'webdav') {
if (!formData.server_url) return null;
let serverUrl = formData.server_url.trim();
// Add https:// if no protocol specified
if (!serverUrl.startsWith('http://') && !serverUrl.startsWith('https://')) {
serverUrl = `https://${serverUrl}`;
}
serverUrl = serverUrl.replace(/\/+$/, ''); // Remove trailing slashes
let webdavPath = '';
if (formData.server_type === 'nextcloud') {
// Nextcloud uses /remote.php/dav/files/{username}
if (!serverUrl.includes('/remote.php/dav/files/')) {
webdavPath = `/remote.php/dav/files/${formData.username || 'username'}`;
}
} else if (formData.server_type === 'owncloud') {
// ownCloud uses /remote.php/webdav
if (!serverUrl.includes('/remote.php/webdav')) {
webdavPath = '/remote.php/webdav';
}
}
// For generic, use the URL as-is
const cleanFolder = firstFolder.replace(/^\/+/, ''); // Remove leading slashes
return {
parts: [
{ text: serverUrl, type: 'server' },
{ text: webdavPath, type: 'path' },
{ text: `/${cleanFolder}`, type: 'folder' },
{ text: `/${exampleFile}`, type: 'file' },
],
};
} else if (formData.source_type === 's3') {
if (!formData.bucket_name) return null;
const endpoint = formData.endpoint_url?.trim() || `https://s3.${formData.region || 'us-east-1'}.amazonaws.com`;
const cleanEndpoint = endpoint.replace(/\/+$/, '');
const prefix = formData.prefix?.trim().replace(/^\/+|\/+$/g, '') || '';
const cleanFolder = firstFolder.replace(/^\/+|\/+$/, '');
const parts: { text: string; type: 'server' | 'path' | 'folder' | 'file' }[] = [
{ text: cleanEndpoint, type: 'server' },
{ text: `/${formData.bucket_name}`, type: 'path' },
{ text: `/${cleanFolder}`, type: 'folder' },
{ text: `/${exampleFile}`, type: 'file' },
];
// Insert prefix after bucket if present
if (prefix) {
parts.splice(2, 0, { text: `/${prefix}`, type: 'path' });
}
return { parts };
} else if (formData.source_type === 'local_folder') {
if (formData.watch_folders.length === 0) return null;
return {
parts: [
{ text: firstFolder, type: 'folder' },
{ text: `/${exampleFile}`, type: 'file' },
],
};
}
return null;
};
// URL Preview Component
const UrlPreviewBox = () => {
const urlParts = buildExampleSyncUrl();
if (!urlParts) return null;
const getColorForType = (type: 'server' | 'path' | 'folder' | 'file') => {
switch (type) {
case 'server': return theme.palette.primary.main;
case 'path': return theme.palette.info.main;
case 'folder': return theme.palette.success.main;
case 'file': return theme.palette.text.secondary;
default: return theme.palette.text.primary;
}
};
const getLabelForType = (type: 'server' | 'path' | 'folder' | 'file') => {
switch (type) {
case 'server': return 'Server URL';
case 'path': return formData.source_type === 'webdav' ? 'WebDAV Path' : 'Bucket/Prefix';
case 'folder': return 'Watch Directory';
case 'file': return 'Example File';
default: return '';
}
};
// Get unique types for legend
const uniqueTypes = Array.from(new Set(urlParts.parts.map(p => p.type)));
return (
<Box
sx={{
mt: 2,
p: 2,
borderRadius: 2,
bgcolor: alpha(theme.palette.background.default, 0.5),
border: `1px solid ${alpha(theme.palette.divider, 0.3)}`,
}}
>
<Typography variant="caption" color="text.secondary" sx={{ display: 'block', mb: 1 }}>
Example sync URL:
</Typography>
<Box
sx={{
fontFamily: 'monospace',
fontSize: '0.85rem',
wordBreak: 'break-all',
p: 1.5,
bgcolor: alpha(theme.palette.common.black, 0.02),
borderRadius: 1,
mb: 1.5,
}}
>
{urlParts.parts.map((part, index) => (
<Box
key={index}
component="span"
sx={{
color: getColorForType(part.type),
fontWeight: part.type === 'folder' ? 600 : 400,
textDecoration: part.type === 'folder' ? 'underline' : 'none',
textDecorationStyle: part.type === 'folder' ? 'dotted' : undefined,
}}
>
{part.text}
</Box>
))}
</Box>
<Stack direction="row" spacing={2} flexWrap="wrap" useFlexGap>
{uniqueTypes.map((type) => (
<Stack key={type} direction="row" alignItems="center" spacing={0.5}>
<Box
sx={{
width: 12,
height: 12,
borderRadius: '50%',
bgcolor: getColorForType(type),
}}
/>
<Typography variant="caption" color="text.secondary">
{getLabelForType(type)}
</Typography>
</Stack>
))}
</Stack>
</Box>
);
};
const handleCreateSource = () => {
setEditingSource(null);
setFormData({
@@ -1635,7 +1797,7 @@ const SourcesPage: React.FC = () => {
Folders to Monitor
</Typography>
</Stack>
<Typography variant="body2" color="text.secondary" sx={{ mb: 2 }}>
Specify which folders to scan for files. Use absolute paths starting with "/".
</Typography>
@@ -1646,14 +1808,14 @@ const SourcesPage: React.FC = () => {
value={newFolder}
onChange={(e) => setNewFolder(e.target.value)}
placeholder="/Documents"
sx={{
sx={{
flexGrow: 1,
'& .MuiOutlinedInput-root': { borderRadius: 2 }
'& .MuiOutlinedInput-root': { borderRadius: 2 }
}}
/>
<Button
variant="outlined"
onClick={addFolder}
<Button
variant="outlined"
onClick={addFolder}
disabled={!newFolder}
sx={{ borderRadius: 2, px: 3 }}
>
@@ -1667,8 +1829,8 @@ const SourcesPage: React.FC = () => {
key={index}
label={folder}
onDelete={() => removeFolder(folder)}
sx={{
mr: 1,
sx={{
mr: 1,
mb: 1,
borderRadius: 2,
bgcolor: alpha(theme.palette.secondary.main, 0.1),
@@ -1678,6 +1840,9 @@ const SourcesPage: React.FC = () => {
))}
</Box>
{/* URL Preview */}
<UrlPreviewBox />
{/* File Extensions */}
<Stack direction="row" alignItems="center" spacing={2} mb={2}>
<Avatar
@@ -1984,7 +2149,7 @@ const SourcesPage: React.FC = () => {
Directories to Monitor
</Typography>
</Stack>
<Typography variant="body2" color="text.secondary" sx={{ mb: 2 }}>
Specify which local directories to scan for files. Use absolute paths.
</Typography>
@@ -1995,14 +2160,14 @@ const SourcesPage: React.FC = () => {
value={newFolder}
onChange={(e) => setNewFolder(e.target.value)}
placeholder="/home/user/Documents"
sx={{
sx={{
flexGrow: 1,
'& .MuiOutlinedInput-root': { borderRadius: 2 }
'& .MuiOutlinedInput-root': { borderRadius: 2 }
}}
/>
<Button
variant="outlined"
onClick={addFolder}
<Button
variant="outlined"
onClick={addFolder}
disabled={!newFolder}
sx={{ borderRadius: 2, px: 3 }}
>
@@ -2016,8 +2181,8 @@ const SourcesPage: React.FC = () => {
key={index}
label={folder}
onDelete={() => removeFolder(folder)}
sx={{
mr: 1,
sx={{
mr: 1,
mb: 1,
borderRadius: 2,
bgcolor: alpha(theme.palette.secondary.main, 0.1),
@@ -2027,10 +2192,13 @@ const SourcesPage: React.FC = () => {
))}
</Box>
{/* URL Preview */}
<UrlPreviewBox />
{/* File Extensions */}
<Stack direction="row" alignItems="center" spacing={2} mb={2}>
<Avatar
sx={{
<Avatar
sx={{
bgcolor: alpha(theme.palette.warning.main, 0.1),
color: theme.palette.warning.main,
width: 32,
@@ -2237,7 +2405,7 @@ const SourcesPage: React.FC = () => {
Object Prefixes to Monitor
</Typography>
</Stack>
<Typography variant="body2" color="text.secondary" sx={{ mb: 2 }}>
Specify which object prefixes (like folders) to scan for files.
</Typography>
@@ -2248,14 +2416,14 @@ const SourcesPage: React.FC = () => {
value={newFolder}
onChange={(e) => setNewFolder(e.target.value)}
placeholder="documents/"
sx={{
sx={{
flexGrow: 1,
'& .MuiOutlinedInput-root': { borderRadius: 2 }
'& .MuiOutlinedInput-root': { borderRadius: 2 }
}}
/>
<Button
variant="outlined"
onClick={addFolder}
<Button
variant="outlined"
onClick={addFolder}
disabled={!newFolder}
sx={{ borderRadius: 2, px: 3 }}
>
@@ -2269,8 +2437,8 @@ const SourcesPage: React.FC = () => {
key={index}
label={folder}
onDelete={() => removeFolder(folder)}
sx={{
mr: 1,
sx={{
mr: 1,
mb: 1,
borderRadius: 2,
bgcolor: alpha(theme.palette.secondary.main, 0.1),
@@ -2280,10 +2448,13 @@ const SourcesPage: React.FC = () => {
))}
</Box>
{/* URL Preview */}
<UrlPreviewBox />
{/* File Extensions */}
<Stack direction="row" alignItems="center" spacing={2} mb={2}>
<Avatar
sx={{
<Avatar
sx={{
bgcolor: alpha(theme.palette.warning.main, 0.1),
color: theme.palette.warning.main,
width: 32,

View File

@@ -0,0 +1,661 @@
import { describe, it, expect } from 'vitest';
/**
* Unit tests for the URL Preview feature in SourcesPage
*
* The URL preview shows users an example of the sync URL that will be constructed
* based on their inputs (server URL, server type, username, watch folders).
*
* This tests the URL construction logic that mirrors what the backend does
* in src/services/webdav/config.rs
*/
// Helper function that mirrors the buildExampleSyncUrl logic from SourcesPage
// Extracted here for direct unit testing
interface UrlPart {
text: string;
type: 'server' | 'path' | 'folder' | 'file';
}
interface FormData {
source_type: 'webdav' | 'local_folder' | 's3';
server_url: string;
username: string;
server_type: 'nextcloud' | 'owncloud' | 'generic';
watch_folders: string[];
bucket_name: string;
region: string;
endpoint_url: string;
prefix: string;
}
function buildExampleSyncUrl(formData: FormData): { parts: UrlPart[] } | null {
const exampleFile = 'document1.pdf';
const firstFolder = formData.watch_folders.length > 0 ? formData.watch_folders[0] : '/Documents';
if (formData.source_type === 'webdav') {
if (!formData.server_url) return null;
let serverUrl = formData.server_url.trim();
// Add https:// if no protocol specified
if (!serverUrl.startsWith('http://') && !serverUrl.startsWith('https://')) {
serverUrl = `https://${serverUrl}`;
}
serverUrl = serverUrl.replace(/\/+$/, ''); // Remove trailing slashes
let webdavPath = '';
if (formData.server_type === 'nextcloud') {
// Nextcloud uses /remote.php/dav/files/{username}
if (!serverUrl.includes('/remote.php/dav/files/')) {
webdavPath = `/remote.php/dav/files/${formData.username || 'username'}`;
}
} else if (formData.server_type === 'owncloud') {
// ownCloud uses /remote.php/webdav
if (!serverUrl.includes('/remote.php/webdav')) {
webdavPath = '/remote.php/webdav';
}
}
// For generic, use the URL as-is
const cleanFolder = firstFolder.replace(/^\/+/, ''); // Remove leading slashes
return {
parts: [
{ text: serverUrl, type: 'server' },
{ text: webdavPath, type: 'path' },
{ text: `/${cleanFolder}`, type: 'folder' },
{ text: `/${exampleFile}`, type: 'file' },
],
};
} else if (formData.source_type === 's3') {
if (!formData.bucket_name) return null;
const endpoint = formData.endpoint_url?.trim() || `https://s3.${formData.region || 'us-east-1'}.amazonaws.com`;
const cleanEndpoint = endpoint.replace(/\/+$/, '');
const prefix = formData.prefix?.trim().replace(/^\/+|\/+$/g, '') || '';
const cleanFolder = firstFolder.replace(/^\/+|\/+$/, '');
const parts: UrlPart[] = [
{ text: cleanEndpoint, type: 'server' },
{ text: `/${formData.bucket_name}`, type: 'path' },
{ text: `/${cleanFolder}`, type: 'folder' },
{ text: `/${exampleFile}`, type: 'file' },
];
// Insert prefix after bucket if present
if (prefix) {
parts.splice(2, 0, { text: `/${prefix}`, type: 'path' });
}
return { parts };
} else if (formData.source_type === 'local_folder') {
if (formData.watch_folders.length === 0) return null;
return {
parts: [
{ text: firstFolder, type: 'folder' },
{ text: `/${exampleFile}`, type: 'file' },
],
};
}
return null;
}
// Helper to join URL parts into a single string for easier assertion
function joinUrlParts(result: { parts: UrlPart[] } | null): string {
if (!result) return '';
return result.parts.map(p => p.text).join('');
}
describe('SourcesPage URL Preview - WebDAV', () => {
const baseWebdavForm: FormData = {
source_type: 'webdav',
server_url: '',
username: '',
server_type: 'generic',
watch_folders: ['/Documents'],
bucket_name: '',
region: 'us-east-1',
endpoint_url: '',
prefix: '',
};
describe('Nextcloud server type', () => {
it('should construct URL with /remote.php/dav/files/{username} path', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://cloud.example.com',
username: 'john',
server_type: 'nextcloud',
watch_folders: ['/Documents'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toBe('https://cloud.example.com/remote.php/dav/files/john/Documents/document1.pdf');
});
it('should use "username" placeholder when username is empty', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://cloud.example.com',
username: '',
server_type: 'nextcloud',
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toContain('/remote.php/dav/files/username/');
});
it('should not duplicate path if URL already contains /remote.php/dav/files/', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://cloud.example.com/remote.php/dav/files/john',
username: 'john',
server_type: 'nextcloud',
};
const result = buildExampleSyncUrl(formData);
// Should not have double /remote.php/dav/files/
expect(result?.parts.filter(p => p.text.includes('/remote.php/dav/files/')).length).toBeLessThanOrEqual(1);
});
it('should handle server URL without trailing slash', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://cloud.example.com',
username: 'john',
server_type: 'nextcloud',
};
const url = joinUrlParts(buildExampleSyncUrl(formData));
// Should not have double slashes (except in https://)
expect(url.replace('https://', '')).not.toContain('//');
});
it('should handle server URL with trailing slash', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://cloud.example.com/',
username: 'john',
server_type: 'nextcloud',
};
const url = joinUrlParts(buildExampleSyncUrl(formData));
// Should not have double slashes
expect(url.replace('https://', '')).not.toContain('//');
});
});
describe('ownCloud server type', () => {
it('should construct URL with /remote.php/webdav path', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://owncloud.example.com',
username: 'john',
server_type: 'owncloud',
watch_folders: ['/Documents'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toBe('https://owncloud.example.com/remote.php/webdav/Documents/document1.pdf');
});
it('should not duplicate path if URL already contains /remote.php/webdav', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://owncloud.example.com/remote.php/webdav',
username: 'john',
server_type: 'owncloud',
};
const result = buildExampleSyncUrl(formData);
// Should not have double /remote.php/webdav
expect(result?.parts.filter(p => p.text.includes('/remote.php/webdav')).length).toBeLessThanOrEqual(1);
});
});
describe('Generic WebDAV server type', () => {
it('should use server URL as-is without adding WebDAV path', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://webdav.example.com/dav',
username: 'john',
server_type: 'generic',
watch_folders: ['/Documents'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toBe('https://webdav.example.com/dav/Documents/document1.pdf');
});
it('should not add /remote.php paths for generic servers', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://custom.webdav.com',
server_type: 'generic',
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).not.toContain('/remote.php');
});
});
describe('Protocol handling', () => {
it('should add https:// when no protocol is specified', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'cloud.example.com',
server_type: 'generic',
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toStartWith('https://');
});
it('should preserve http:// when explicitly specified', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'http://local.webdav.com',
server_type: 'generic',
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toStartWith('http://');
expect(url).not.toStartWith('https://');
});
it('should preserve https:// when explicitly specified', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://secure.webdav.com',
server_type: 'generic',
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toStartWith('https://');
});
});
describe('Watch folder handling', () => {
it('should use first watch folder in the URL', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://webdav.example.com',
server_type: 'generic',
watch_folders: ['/Photos', '/Documents', '/Videos'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toContain('/Photos/');
expect(url).not.toContain('/Documents/');
});
it('should handle watch folder with leading slash', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://webdav.example.com',
server_type: 'generic',
watch_folders: ['/Documents'],
};
const url = joinUrlParts(buildExampleSyncUrl(formData));
// Should not have double slashes around folder
expect(url).toContain('/Documents/');
expect(url.replace('https://', '')).not.toContain('//');
});
it('should handle watch folder without leading slash', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://webdav.example.com',
server_type: 'generic',
watch_folders: ['Documents'],
};
const url = joinUrlParts(buildExampleSyncUrl(formData));
expect(url).toContain('/Documents/');
});
it('should default to /Documents when watch_folders is empty', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: 'https://webdav.example.com',
server_type: 'generic',
watch_folders: [],
};
const url = joinUrlParts(buildExampleSyncUrl(formData));
expect(url).toContain('/Documents/');
});
});
describe('Edge cases', () => {
it('should return null when server_url is empty', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: '',
};
const result = buildExampleSyncUrl(formData);
expect(result).toBeNull();
});
it('should handle whitespace in server_url', () => {
const formData: FormData = {
...baseWebdavForm,
server_url: ' https://webdav.example.com ',
server_type: 'generic',
};
const url = joinUrlParts(buildExampleSyncUrl(formData));
expect(url).toStartWith('https://');
expect(url).not.toContain(' ');
});
});
});
describe('SourcesPage URL Preview - S3', () => {
const baseS3Form: FormData = {
source_type: 's3',
server_url: '',
username: '',
server_type: 'generic',
watch_folders: ['/documents'],
bucket_name: '',
region: 'us-east-1',
endpoint_url: '',
prefix: '',
};
describe('AWS S3', () => {
it('should construct URL with default AWS endpoint when endpoint_url is empty', () => {
const formData: FormData = {
...baseS3Form,
bucket_name: 'my-bucket',
region: 'us-west-2',
watch_folders: ['/documents'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toBe('https://s3.us-west-2.amazonaws.com/my-bucket/documents/document1.pdf');
});
it('should use us-east-1 as default region', () => {
const formData: FormData = {
...baseS3Form,
bucket_name: 'my-bucket',
region: '',
watch_folders: ['/documents'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toContain('s3.us-east-1.amazonaws.com');
});
});
describe('S3-compatible storage (MinIO)', () => {
it('should use custom endpoint_url when provided', () => {
const formData: FormData = {
...baseS3Form,
bucket_name: 'my-bucket',
endpoint_url: 'https://minio.example.com',
watch_folders: ['/documents'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toBe('https://minio.example.com/my-bucket/documents/document1.pdf');
});
it('should handle endpoint_url with trailing slash', () => {
const formData: FormData = {
...baseS3Form,
bucket_name: 'my-bucket',
endpoint_url: 'https://minio.example.com/',
watch_folders: ['/documents'],
};
const url = joinUrlParts(buildExampleSyncUrl(formData));
// Should not have double slashes
expect(url.replace('https://', '')).not.toContain('//');
});
});
describe('Prefix handling', () => {
it('should include prefix in URL when provided', () => {
const formData: FormData = {
...baseS3Form,
bucket_name: 'my-bucket',
prefix: 'uploads/2024',
watch_folders: ['/documents'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toContain('/uploads/2024/');
});
it('should handle prefix with leading/trailing slashes', () => {
const formData: FormData = {
...baseS3Form,
bucket_name: 'my-bucket',
prefix: '/uploads/2024/',
watch_folders: ['/documents'],
};
const url = joinUrlParts(buildExampleSyncUrl(formData));
// Should normalize slashes
expect(url.replace('https://', '')).not.toContain('//');
});
it('should not include prefix segment when prefix is empty', () => {
const formData: FormData = {
...baseS3Form,
bucket_name: 'my-bucket',
prefix: '',
watch_folders: ['/documents'],
};
const result = buildExampleSyncUrl(formData);
// Should only have 4 parts: server, bucket, folder, file (no prefix)
expect(result?.parts.length).toBe(4);
});
});
describe('Part types', () => {
it('should correctly type each URL part', () => {
const formData: FormData = {
...baseS3Form,
bucket_name: 'my-bucket',
endpoint_url: 'https://s3.example.com',
prefix: 'prefix',
watch_folders: ['/documents'],
};
const result = buildExampleSyncUrl(formData);
expect(result?.parts[0].type).toBe('server'); // endpoint
expect(result?.parts[1].type).toBe('path'); // bucket
expect(result?.parts[2].type).toBe('path'); // prefix
expect(result?.parts[3].type).toBe('folder'); // watch folder
expect(result?.parts[4].type).toBe('file'); // example file
});
});
describe('Edge cases', () => {
it('should return null when bucket_name is empty', () => {
const formData: FormData = {
...baseS3Form,
bucket_name: '',
};
const result = buildExampleSyncUrl(formData);
expect(result).toBeNull();
});
});
});
describe('SourcesPage URL Preview - Local Folder', () => {
const baseLocalForm: FormData = {
source_type: 'local_folder',
server_url: '',
username: '',
server_type: 'generic',
watch_folders: [],
bucket_name: '',
region: 'us-east-1',
endpoint_url: '',
prefix: '',
};
it('should show local path with example file', () => {
const formData: FormData = {
...baseLocalForm,
watch_folders: ['/home/user/Documents'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toBe('/home/user/Documents/document1.pdf');
});
it('should use first watch folder', () => {
const formData: FormData = {
...baseLocalForm,
watch_folders: ['/var/data', '/home/user/Documents'],
};
const result = buildExampleSyncUrl(formData);
const url = joinUrlParts(result);
expect(url).toContain('/var/data/');
expect(url).not.toContain('/home/user/Documents');
});
it('should return null when watch_folders is empty', () => {
const formData: FormData = {
...baseLocalForm,
watch_folders: [],
};
const result = buildExampleSyncUrl(formData);
expect(result).toBeNull();
});
it('should correctly type parts for local folder', () => {
const formData: FormData = {
...baseLocalForm,
watch_folders: ['/home/user/Documents'],
};
const result = buildExampleSyncUrl(formData);
expect(result?.parts.length).toBe(2);
expect(result?.parts[0].type).toBe('folder');
expect(result?.parts[1].type).toBe('file');
});
});
describe('SourcesPage URL Preview - URL Part Types', () => {
it('WebDAV should have correct part types', () => {
const formData: FormData = {
source_type: 'webdav',
server_url: 'https://cloud.example.com',
username: 'john',
server_type: 'nextcloud',
watch_folders: ['/Documents'],
bucket_name: '',
region: 'us-east-1',
endpoint_url: '',
prefix: '',
};
const result = buildExampleSyncUrl(formData);
expect(result?.parts[0].type).toBe('server'); // https://cloud.example.com
expect(result?.parts[1].type).toBe('path'); // /remote.php/dav/files/john
expect(result?.parts[2].type).toBe('folder'); // /Documents
expect(result?.parts[3].type).toBe('file'); // /document1.pdf
});
it('Generic WebDAV should have empty path part', () => {
const formData: FormData = {
source_type: 'webdav',
server_url: 'https://webdav.example.com',
username: 'john',
server_type: 'generic',
watch_folders: ['/Documents'],
bucket_name: '',
region: 'us-east-1',
endpoint_url: '',
prefix: '',
};
const result = buildExampleSyncUrl(formData);
// For generic, webdavPath is empty string
expect(result?.parts[1].text).toBe('');
expect(result?.parts[1].type).toBe('path');
});
});
// Custom matcher for startsWith
expect.extend({
toStartWith(received: string, expected: string) {
const pass = received.startsWith(expected);
return {
message: () =>
pass
? `expected ${received} not to start with ${expected}`
: `expected ${received} to start with ${expected}`,
pass,
};
},
});
declare global {
namespace jest {
interface Matchers<R> {
toStartWith(expected: string): R;
}
}
}