uri.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. // Copyright 2017 The go-ethereum Authors
  2. // This file is part of the go-ethereum library.
  3. //
  4. // The go-ethereum library is free software: you can redistribute it and/or modify
  5. // it under the terms of the GNU Lesser General Public License as published by
  6. // the Free Software Foundation, either version 3 of the License, or
  7. // (at your option) any later version.
  8. //
  9. // The go-ethereum library is distributed in the hope that it will be useful,
  10. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. // GNU Lesser General Public License for more details.
  13. //
  14. // You should have received a copy of the GNU Lesser General Public License
  15. // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
  16. package api
  17. import (
  18. "fmt"
  19. "net/url"
  20. "regexp"
  21. "strings"
  22. "github.com/ethereum/go-ethereum/common"
  23. "github.com/ethereum/go-ethereum/swarm/storage"
  24. )
  25. //matches hex swarm hashes
  26. // TODO: this is bad, it should not be hardcoded how long is a hash
  27. var hashMatcher = regexp.MustCompile("^([0-9A-Fa-f]{64})([0-9A-Fa-f]{64})?$")
  28. // URI is a reference to content stored in swarm.
  29. type URI struct {
  30. // Scheme has one of the following values:
  31. //
  32. // * bzz - an entry in a swarm manifest
  33. // * bzz-raw - raw swarm content
  34. // * bzz-immutable - immutable URI of an entry in a swarm manifest
  35. // (address is not resolved)
  36. // * bzz-list - list of all files contained in a swarm manifest
  37. //
  38. Scheme string
  39. // Addr is either a hexadecimal storage address or it an address which
  40. // resolves to a storage address
  41. Addr string
  42. // addr stores the parsed storage address
  43. addr storage.Address
  44. // Path is the path to the content within a swarm manifest
  45. Path string
  46. }
  47. // Parse parses rawuri into a URI struct, where rawuri is expected to have one
  48. // of the following formats:
  49. //
  50. // * <scheme>:/
  51. // * <scheme>:/<addr>
  52. // * <scheme>:/<addr>/<path>
  53. // * <scheme>://
  54. // * <scheme>://<addr>
  55. // * <scheme>://<addr>/<path>
  56. //
  57. // with scheme one of bzz, bzz-raw, bzz-immutable, bzz-list or bzz-hash
  58. func Parse(rawuri string) (*URI, error) {
  59. u, err := url.Parse(rawuri)
  60. if err != nil {
  61. return nil, err
  62. }
  63. uri := &URI{Scheme: u.Scheme}
  64. // check the scheme is valid
  65. switch uri.Scheme {
  66. case "bzz", "bzz-raw", "bzz-immutable", "bzz-list", "bzz-hash", "bzz-resource":
  67. default:
  68. return nil, fmt.Errorf("unknown scheme %q", u.Scheme)
  69. }
  70. // handle URIs like bzz://<addr>/<path> where the addr and path
  71. // have already been split by url.Parse
  72. if u.Host != "" {
  73. uri.Addr = u.Host
  74. uri.Path = strings.TrimLeft(u.Path, "/")
  75. return uri, nil
  76. }
  77. // URI is like bzz:/<addr>/<path> so split the addr and path from
  78. // the raw path (which will be /<addr>/<path>)
  79. parts := strings.SplitN(strings.TrimLeft(u.Path, "/"), "/", 2)
  80. uri.Addr = parts[0]
  81. if len(parts) == 2 {
  82. uri.Path = parts[1]
  83. }
  84. return uri, nil
  85. }
  86. func (u *URI) Resource() bool {
  87. return u.Scheme == "bzz-resource"
  88. }
  89. func (u *URI) Raw() bool {
  90. return u.Scheme == "bzz-raw"
  91. }
  92. func (u *URI) Immutable() bool {
  93. return u.Scheme == "bzz-immutable"
  94. }
  95. func (u *URI) List() bool {
  96. return u.Scheme == "bzz-list"
  97. }
  98. func (u *URI) Hash() bool {
  99. return u.Scheme == "bzz-hash"
  100. }
  101. func (u *URI) String() string {
  102. return u.Scheme + ":/" + u.Addr + "/" + u.Path
  103. }
  104. func (u *URI) Address() storage.Address {
  105. if u.addr != nil {
  106. return u.addr
  107. }
  108. if hashMatcher.MatchString(u.Addr) {
  109. u.addr = common.Hex2Bytes(u.Addr)
  110. return u.addr
  111. }
  112. return nil
  113. }