Skip to content

Commit

Permalink
feat: add p/moul/realmpath (gnolang#3094)
Browse files Browse the repository at this point in the history
Lightweight `Render.path` parsing and link generation library with an
idiomatic API, closely resembling that of `net/url`.

---------

Signed-off-by: moul <[email protected]>
  • Loading branch information
moul authored and r3v4s committed Dec 10, 2024
1 parent e0d2331 commit c9b19c8
Show file tree
Hide file tree
Showing 3 changed files with 257 additions and 0 deletions.
6 changes: 6 additions & 0 deletions examples/gno.land/p/moul/realmpath/gno.mod
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
module gno.land/p/moul/realmpath

require (
gno.land/p/demo/uassert v0.0.0-latest
gno.land/p/demo/urequire v0.0.0-latest
)
100 changes: 100 additions & 0 deletions examples/gno.land/p/moul/realmpath/realmpath.gno
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
// Package realmpath is a lightweight Render.path parsing and link generation
// library with an idiomatic API, closely resembling that of net/url.
//
// This package provides utilities for parsing request paths and query
// parameters, allowing you to extract path segments and manipulate query
// values.
//
// Example usage:
//
// import "gno.land/p/moul/realmpath"
//
// func Render(path string) string {
// // Parsing a sample path with query parameters
// path = "hello/world?foo=bar&baz=foobar"
// req := realmpath.Parse(path)
//
// // Accessing parsed path and query parameters
// println(req.Path) // Output: hello/world
// println(req.PathPart(0)) // Output: hello
// println(req.PathPart(1)) // Output: world
// println(req.Query.Get("foo")) // Output: bar
// println(req.Query.Get("baz")) // Output: foobar
//
// // Rebuilding the URL
// println(req.String()) // Output: /r/current/realm:hello/world?baz=foobar&foo=bar
// }
package realmpath

import (
"net/url"
"std"
"strings"
)

const chainDomain = "gno.land" // XXX: std.ChainDomain (#2911)

// Request represents a parsed request.
type Request struct {
Path string // The path of the request
Query url.Values // The parsed query parameters
Realm string // The realm associated with the request
}

// Parse takes a raw path string and returns a Request object.
// It splits the path into its components and parses any query parameters.
func Parse(rawPath string) *Request {
// Split the raw path into path and query components
path, query := splitPathAndQuery(rawPath)

// Parse the query string into url.Values
queryValues, _ := url.ParseQuery(query)

return &Request{
Path: path, // Set the path
Query: queryValues, // Set the parsed query values
}
}

// PathParts returns the segments of the path as a slice of strings.
// It trims leading and trailing slashes and splits the path by slashes.
func (r *Request) PathParts() []string {
return strings.Split(strings.Trim(r.Path, "/"), "/")
}

// PathPart returns the specified part of the path.
// If the index is out of bounds, it returns an empty string.
func (r *Request) PathPart(index int) string {
parts := r.PathParts() // Get the path segments
if index < 0 || index >= len(parts) {
return "" // Return empty if index is out of bounds
}
return parts[index] // Return the specified path part
}

// String rebuilds the URL from the path and query values.
// If the Realm is not set, it automatically retrieves the current realm path.
func (r *Request) String() string {
// Automatically set the Realm if it is not already defined
if r.Realm == "" {
r.Realm = std.CurrentRealm().PkgPath() // Get the current realm path
}

// Rebuild the path using the realm and path parts
relativePkgPath := strings.TrimPrefix(r.Realm, chainDomain) // Trim the chain domain prefix
reconstructedPath := relativePkgPath + ":" + strings.Join(r.PathParts(), "/")

// Rebuild the query string
queryString := r.Query.Encode() // Encode the query parameters
if queryString != "" {
return reconstructedPath + "?" + queryString // Return the full URL with query
}
return reconstructedPath // Return the path without query parameters
}

func splitPathAndQuery(rawPath string) (string, string) {
if idx := strings.Index(rawPath, "?"); idx != -1 {
return rawPath[:idx], rawPath[idx+1:] // Split at the first '?' found
}
return rawPath, "" // No query string present
}
151 changes: 151 additions & 0 deletions examples/gno.land/p/moul/realmpath/realmpath_test.gno
Original file line number Diff line number Diff line change
@@ -0,0 +1,151 @@
package realmpath_test

import (
"net/url"
"std"
"testing"

"gno.land/p/demo/uassert"
"gno.land/p/demo/urequire"
"gno.land/p/moul/realmpath"
)

func TestExample(t *testing.T) {
std.TestSetRealm(std.NewCodeRealm("gno.land/r/lorem/ipsum"))

// initial parsing
path := "hello/world?foo=bar&baz=foobar"
req := realmpath.Parse(path)
urequire.False(t, req == nil, "req should not be nil")
uassert.Equal(t, req.Path, "hello/world")
uassert.Equal(t, req.Query.Get("foo"), "bar")
uassert.Equal(t, req.Query.Get("baz"), "foobar")
uassert.Equal(t, req.String(), "/r/lorem/ipsum:hello/world?baz=foobar&foo=bar")

// alter query
req.Query.Set("hey", "salut")
uassert.Equal(t, req.String(), "/r/lorem/ipsum:hello/world?baz=foobar&foo=bar&hey=salut")

// alter path
req.Path = "bye/ciao"
uassert.Equal(t, req.String(), "/r/lorem/ipsum:bye/ciao?baz=foobar&foo=bar&hey=salut")
}

func TestParse(t *testing.T) {
std.TestSetRealm(std.NewCodeRealm("gno.land/r/lorem/ipsum"))

tests := []struct {
rawPath string
realm string // optional
expectedPath string
expectedQuery url.Values
expectedString string
}{
{
rawPath: "hello/world?foo=bar&baz=foobar",
expectedPath: "hello/world",
expectedQuery: url.Values{
"foo": []string{"bar"},
"baz": []string{"foobar"},
},
expectedString: "/r/lorem/ipsum:hello/world?baz=foobar&foo=bar",
},
{
rawPath: "api/v1/resource?search=test&limit=10",
expectedPath: "api/v1/resource",
expectedQuery: url.Values{
"search": []string{"test"},
"limit": []string{"10"},
},
expectedString: "/r/lorem/ipsum:api/v1/resource?limit=10&search=test",
},
{
rawPath: "singlepath",
expectedPath: "singlepath",
expectedQuery: url.Values{},
expectedString: "/r/lorem/ipsum:singlepath",
},
{
rawPath: "path/with/trailing/slash/",
expectedPath: "path/with/trailing/slash/",
expectedQuery: url.Values{},
expectedString: "/r/lorem/ipsum:path/with/trailing/slash",
},
{
rawPath: "emptyquery?",
expectedPath: "emptyquery",
expectedQuery: url.Values{},
expectedString: "/r/lorem/ipsum:emptyquery",
},
{
rawPath: "path/with/special/characters/?key=val%20ue&anotherKey=with%21special%23chars",
expectedPath: "path/with/special/characters/",
expectedQuery: url.Values{
"key": []string{"val ue"},
"anotherKey": []string{"with!special#chars"},
},
expectedString: "/r/lorem/ipsum:path/with/special/characters?anotherKey=with%21special%23chars&key=val+ue",
},
{
rawPath: "path/with/empty/key?keyEmpty&=valueEmpty",
expectedPath: "path/with/empty/key",
expectedQuery: url.Values{
"keyEmpty": []string{""},
"": []string{"valueEmpty"},
},
expectedString: "/r/lorem/ipsum:path/with/empty/key?=valueEmpty&keyEmpty=",
},
{
rawPath: "path/with/multiple/empty/keys?=empty1&=empty2",
expectedPath: "path/with/multiple/empty/keys",
expectedQuery: url.Values{
"": []string{"empty1", "empty2"},
},
expectedString: "/r/lorem/ipsum:path/with/multiple/empty/keys?=empty1&=empty2",
},
{
rawPath: "path/with/percent-encoded/%20space?query=hello%20world",
expectedPath: "path/with/percent-encoded/%20space", // XXX: should we decode?
expectedQuery: url.Values{
"query": []string{"hello world"},
},
expectedString: "/r/lorem/ipsum:path/with/percent-encoded/%20space?query=hello+world",
},
{
rawPath: "path/with/very/long/query?key1=value1&key2=value2&key3=value3&key4=value4&key5=value5&key6=value6",
expectedPath: "path/with/very/long/query",
expectedQuery: url.Values{
"key1": []string{"value1"},
"key2": []string{"value2"},
"key3": []string{"value3"},
"key4": []string{"value4"},
"key5": []string{"value5"},
"key6": []string{"value6"},
},
expectedString: "/r/lorem/ipsum:path/with/very/long/query?key1=value1&key2=value2&key3=value3&key4=value4&key5=value5&key6=value6",
},
{
rawPath: "custom/realm?foo=bar&baz=foobar",
realm: "gno.land/r/foo/bar",
expectedPath: "custom/realm",
expectedQuery: url.Values{
"foo": []string{"bar"},
"baz": []string{"foobar"},
},
expectedString: "/r/foo/bar:custom/realm?baz=foobar&foo=bar",
},
}

for _, tt := range tests {
t.Run(tt.rawPath, func(t *testing.T) {
req := realmpath.Parse(tt.rawPath)
req.Realm = tt.realm // set optional realm
urequire.False(t, req == nil, "req should not be nil")
uassert.Equal(t, req.Path, tt.expectedPath)
urequire.Equal(t, len(req.Query), len(tt.expectedQuery))
uassert.Equal(t, req.Query.Encode(), tt.expectedQuery.Encode())
// XXX: uassert.Equal(t, req.Query, tt.expectedQuery)
uassert.Equal(t, req.String(), tt.expectedString)
})
}
}

0 comments on commit c9b19c8

Please sign in to comment.