@@ -10,7 +10,7 @@ import { CID } from 'multiformats/cid'
10
10
import { base32 } from 'multiformats/bases/base32'
11
11
import { base36 } from 'multiformats/bases/base36'
12
12
13
- // Different from raw JSON. Determenistic
13
+ // Different from raw JSON. Deterministic
14
14
import * as dagJSON from '@ipld/dag-json'
15
15
import * as cbor from '@ipld/dag-cbor'
16
16
@@ -147,7 +147,8 @@ export default function makeIPFSFetch ({
147
147
// TODO: better status?
148
148
status : 400 ,
149
149
headers : defaultHeaders ,
150
- body : 'Unsupproted content-type, must be dag-cbor, application/json, or dag-json'
150
+ body :
151
+ 'Unsupported content-type, must be dag-cbor, application/json, or dag-json'
151
152
}
152
153
}
153
154
@@ -199,7 +200,8 @@ export default function makeIPFSFetch ({
199
200
// TODO: better status?
200
201
status : 400 ,
201
202
headers : defaultHeaders ,
202
- body : 'Unsupproted content-type, must be dag-cbor, application/json, or dag-json'
203
+ body :
204
+ 'Unsupported content-type, must be dag-cbor, application/json, or dag-json'
203
205
}
204
206
}
205
207
@@ -284,7 +286,7 @@ export default function makeIPFSFetch ({
284
286
const { url, signal } = request
285
287
const topic = new URL ( url ) . hostname
286
288
const payload = await request . arrayBuffer ( )
287
- // TODO: Handle oversized messages wihth 413
289
+ // TODO: Handle oversized messages with 413
288
290
await ipfs . pubsub . publish ( topic , payload , {
289
291
signal,
290
292
timeout
@@ -318,6 +320,7 @@ export default function makeIPFSFetch ({
318
320
}
319
321
}
320
322
} )
323
+
321
324
// TODO: Generate from headers somehow
322
325
router . post ( `ipns://${ SPECIAL_HOSTNAME } /` , async ( { url, signal } ) => {
323
326
const key = new URL ( url ) . searchParams . get ( 'key' )
@@ -333,9 +336,11 @@ export default function makeIPFSFetch ({
333
336
status : 201 ,
334
337
headers : {
335
338
Location : keyURL
336
- }
339
+ } ,
340
+ body : keyURL
337
341
}
338
342
} )
343
+
339
344
router . delete ( `ipns://${ SPECIAL_HOSTNAME } /` , async ( { url, signal } ) => {
340
345
const key = new URL ( url ) . searchParams . get ( 'key' )
341
346
await ipfs . key . rm ( key , {
@@ -417,7 +422,7 @@ export default function makeIPFSFetch ({
417
422
const contentType = headers . get ( 'Content-Type' ) || ''
418
423
const isFormData = contentType . includes ( 'multipart/form-data' )
419
424
420
- const ipnsPath = urlToIPFSPath ( url )
425
+ const ipnsPath = urlToIPNSPath ( url )
421
426
const split = ipnsPath . split ( '/' )
422
427
const keyName = split [ 2 ]
423
428
const subpath = split . slice ( 3 ) . join ( '/' )
@@ -445,7 +450,9 @@ export default function makeIPFSFetch ({
445
450
const { hostname : keyName } = new URL ( url )
446
451
447
452
const rawValue = await request . text ( )
448
- const value = rawValue . replace ( / ^ i p f s : \/ \/ / , '/ipfs/' ) . replace ( / ^ i p n s : \/ \/ / , '/ipns/' )
453
+ const value = rawValue
454
+ . replace ( / ^ i p f s : \/ \/ / , '/ipfs/' )
455
+ . replace ( / ^ i p n s : \/ \/ / , '/ipns/' )
449
456
450
457
return updateIPNS ( keyName , value , signal )
451
458
} )
@@ -462,7 +469,9 @@ export default function makeIPFSFetch ({
462
469
const ipfsPath = await resolveIPNS ( ipnsPath , signal )
463
470
const { body : updatedURL } = await deleteData ( ipfsPath , signal )
464
471
465
- const value = updatedURL . replace ( / ^ i p f s : \/ \/ / , '/ipfs/' ) . replace ( / ^ i p n s : \/ \/ / , '/ipns/' )
472
+ const value = updatedURL
473
+ . replace ( / ^ i p f s : \/ \/ / , '/ipfs/' )
474
+ . replace ( / ^ i p n s : \/ \/ / , '/ipns/' )
466
475
467
476
return updateIPNS ( keyName , value , signal )
468
477
} )
@@ -505,24 +514,30 @@ export default function makeIPFSFetch ({
505
514
const accept = reqHeaders . get ( 'Accept' ) || ''
506
515
const expectedType = format || accept
507
516
508
- const headers = { ...defaultHeaders }
517
+ const headersResponse = { ...defaultHeaders }
509
518
510
- if ( expectedType === 'raw' || expectedType === 'application/vnd.ipld.raw' ) {
519
+ if (
520
+ expectedType === 'raw' ||
521
+ expectedType === 'application/vnd.ipld.raw'
522
+ ) {
511
523
const body = await ipfs . block . get ( ipfsPath , {
512
524
timeout,
513
525
signal
514
526
} )
515
527
516
- headers [ 'Content-Type' ] = 'application/vnd.ipld.raw'
528
+ headersResponse [ 'Content-Type' ] = 'application/vnd.ipld.raw'
517
529
518
530
return {
519
531
status : 200 ,
520
- headers,
532
+ headers : headersResponse ,
521
533
body
522
534
}
523
535
}
524
536
525
- if ( expectedType === 'car' || expectedType === 'application/vnd.ipld.car' ) {
537
+ if (
538
+ expectedType === 'car' ||
539
+ expectedType === 'application/vnd.ipld.car'
540
+ ) {
526
541
const { cid } = await ipfs . dag . resolve ( ipfsPath , {
527
542
timeout,
528
543
signal
@@ -533,11 +548,11 @@ export default function makeIPFSFetch ({
533
548
signal
534
549
} )
535
550
536
- headers [ 'Content-Type' ] = 'application/vnd.ipld.car'
551
+ headersResponse [ 'Content-Type' ] = 'application/vnd.ipld.car'
537
552
538
553
return {
539
554
status : 200 ,
540
- headers,
555
+ headers : headersResponse ,
541
556
body
542
557
}
543
558
}
@@ -551,34 +566,49 @@ export default function makeIPFSFetch ({
551
566
552
567
try {
553
568
const stats = await collect ( ipfs . ls ( ipfsPath , { signal, timeout } ) )
554
- const files = stats . map ( ( { name, type } ) => ( type === 'dir' ) ? `${ name } /` : name )
569
+ // Decode path segments for accurate file names
570
+ const files = stats . map ( ( { name, type } ) =>
571
+ type === 'dir' ? `${ decodeURIComponent ( name ) } /` : decodeURIComponent ( name )
572
+ )
555
573
556
574
if ( files . includes ( 'index.html' ) ) {
557
575
if ( ! searchParams . has ( 'noResolve' ) ) {
558
- return serveFile ( posixPath . join ( ipfsPath , 'index.html' ) , searchParams , reqHeaders , headers , signal )
576
+ return serveFile (
577
+ posixPath . join ( ipfsPath , 'index.html' ) ,
578
+ searchParams ,
579
+ reqHeaders ,
580
+ headersResponse ,
581
+ signal
582
+ )
559
583
}
560
584
}
561
585
562
586
if ( accept . includes ( 'text/html' ) ) {
563
- const page = await renderIndex ( url , files , fetch )
564
- headers [ 'Content-Type' ] = 'text/html; charset=utf-8'
587
+ // Encode file names to handle spaces and special characters
588
+ const encodedFiles = files . map ( ( file ) =>
589
+ file . endsWith ( '/' )
590
+ ? `${ encodeURIComponent ( file . slice ( 0 , - 1 ) ) } /`
591
+ : encodeURIComponent ( file )
592
+ )
593
+ const page = await renderIndex ( url , encodedFiles , fetch )
594
+ headersResponse [ 'Content-Type' ] = 'text/html; charset=utf-8'
565
595
body = page
566
596
} else {
567
597
const json = JSON . stringify ( files , null , '\t' )
568
- headers [ 'Content-Type' ] = `${ MIME_JSON } ; charset=utf-8`
598
+ headersResponse [ 'Content-Type' ] = `${ MIME_JSON } ; charset=utf-8`
569
599
body = json
570
600
}
571
601
572
602
return {
573
603
status : 200 ,
574
- headers,
604
+ headers : headersResponse ,
575
605
body
576
606
}
577
607
} catch {
578
- return serveFile ( ipfsPath , searchParams , reqHeaders , headers , signal )
608
+ return serveFile ( ipfsPath , searchParams , reqHeaders , headersResponse , signal )
579
609
}
580
610
} else {
581
- return serveFile ( ipfsPath , searchParams , reqHeaders , headers , signal )
611
+ return serveFile ( ipfsPath , searchParams , reqHeaders , headersResponse , signal )
582
612
}
583
613
}
584
614
@@ -590,11 +620,10 @@ export default function makeIPFSFetch ({
590
620
if ( stat . type === 'directory' ) {
591
621
// TODO: Something for directories?
592
622
if ( ! noResolve ) {
593
- const stats = await collect ( ipfs . ls ( ipfsPath , {
594
- signal,
595
- timeout
596
- } ) )
597
- const files = stats . map ( ( { name, type } ) => ( type === 'dir' ) ? `${ name } /` : name )
623
+ const stats = await collect ( ipfs . ls ( ipfsPath , { signal, timeout } ) )
624
+ const files = stats . map ( ( { name, type } ) =>
625
+ type === 'dir' ? `${ decodeURIComponent ( name ) } /` : decodeURIComponent ( name )
626
+ )
598
627
if ( files . includes ( 'index.html' ) ) {
599
628
ipfsPath = posixPath . join ( ipfsPath , 'index.html' )
600
629
} else {
@@ -636,7 +665,7 @@ export default function makeIPFSFetch ({
636
665
const ranges = parseRange ( size , isRanged )
637
666
if ( ranges && ranges . length && ranges . type === 'bytes' ) {
638
667
const [ { start, end } ] = ranges
639
- const length = ( end - start + 1 )
668
+ const length = end - start + 1
640
669
headers [ 'Content-Length' ] = `${ length } `
641
670
headers [ 'Content-Range' ] = `bytes ${ start } -${ end } /${ size } `
642
671
return {
@@ -731,7 +760,9 @@ export default function makeIPFSFetch ({
731
760
return keys . find ( ( { name, id } ) => {
732
761
if ( name === keyName ) return true
733
762
try {
734
- return ( CID . parse ( id , bases ) . toV1 ( ) . toString ( base36 ) === keyName )
763
+ return (
764
+ CID . parse ( id , bases ) . toV1 ( ) . toString ( base36 ) === keyName
765
+ )
735
766
} catch {
736
767
return false
737
768
}
@@ -777,7 +808,7 @@ export default function makeIPFSFetch ({
777
808
}
778
809
}
779
810
780
- async function uploadData ( ipfsPath , response , isFormData , signal ) {
811
+ async function uploadData ( ipfsPath , request , isFormData , signal ) {
781
812
const tmpDir = makeTmpDir ( )
782
813
const { rootCID, relativePath } = cidFromPath ( ipfsPath )
783
814
@@ -791,16 +822,16 @@ export default function makeIPFSFetch ({
791
822
}
792
823
793
824
if ( isFormData ) {
794
- const formData = await response . formData ( )
825
+ const formData = await request . formData ( )
795
826
const toWait = [ ]
796
827
797
828
for ( const [ fieldName , fileData ] of formData ) {
798
- // TODO: Should we filter by field name?
829
+ // Filter by field name if necessary
799
830
if ( fieldName !== 'file' ) continue
800
- // Must not be a file
831
+ // Must have a filename
801
832
if ( ! fileData . name ) continue
802
833
const fileName = fileData . name
803
- const finalPath = posixPath . join ( tmpDir , relativePath , fileName )
834
+ const finalPath = posixPath . join ( tmpDir , relativePath , encodeURIComponent ( fileName ) )
804
835
const result = ipfs . files . write ( finalPath , fileData , {
805
836
cidVersion : 1 ,
806
837
parents : true ,
@@ -815,9 +846,12 @@ export default function makeIPFSFetch ({
815
846
816
847
await Promise . all ( toWait )
817
848
} else {
818
- const path = posixPath . join ( tmpDir , ensureStartingSlash ( stripEndingSlash ( relativePath ) ) )
849
+ const path = posixPath . join (
850
+ tmpDir ,
851
+ ensureStartingSlash ( stripEndingSlash ( relativePath ) )
852
+ )
819
853
820
- await ipfs . files . write ( path , await response . blob ( ) , {
854
+ await ipfs . files . write ( path , await request . blob ( ) , {
821
855
signal,
822
856
parents : true ,
823
857
truncate : true ,
@@ -832,7 +866,13 @@ export default function makeIPFSFetch ({
832
866
833
867
const cidHash = cid . toString ( )
834
868
const endPath = isFormData ? relativePath : stripEndingSlash ( relativePath )
835
- const addedURL = `ipfs://${ cidHash } ${ ensureStartingSlash ( endPath ) } `
869
+ const encodedEndPath = isFormData
870
+ ? relativePath
871
+ . split ( '/' )
872
+ . map ( ( segment ) => encodeURIComponent ( segment ) )
873
+ . join ( '/' )
874
+ : encodeURIComponent ( endPath )
875
+ const addedURL = `ipfs://${ cidHash } ${ ensureStartingSlash ( encodedEndPath ) } `
836
876
837
877
return addedURL
838
878
}
@@ -909,9 +949,24 @@ function keyToURL ({ id }) {
909
949
910
950
function urlToIPFSPath ( url ) {
911
951
const { pathname, hostname } = new URL ( url )
912
- return `/ipfs/${ hostname } ${ pathname } `
952
+ const decodedPathSegments = pathname
953
+ . split ( '/' )
954
+ . filter ( Boolean )
955
+ . map ( ( part ) => decodeURIComponent ( part ) )
956
+ const encodedPathSegments = decodedPathSegments . map ( ( part ) =>
957
+ encodeURIComponent ( part )
958
+ )
959
+ return `/ipfs/${ hostname } /${ encodedPathSegments . join ( '/' ) } `
913
960
}
961
+
914
962
function urlToIPNSPath ( url ) {
915
963
const { pathname, hostname } = new URL ( url )
916
- return `/ipns/${ hostname } ${ pathname } `
964
+ const decodedPathSegments = pathname
965
+ . split ( '/' )
966
+ . filter ( Boolean )
967
+ . map ( ( part ) => decodeURIComponent ( part ) )
968
+ const encodedPathSegments = decodedPathSegments . map ( ( part ) =>
969
+ encodeURIComponent ( part )
970
+ )
971
+ return `/ipns/${ hostname } /${ encodedPathSegments . join ( '/' ) } `
917
972
}
0 commit comments