From f4e7643e227ba5261adbdf6cafe7db30e36cf297 Mon Sep 17 00:00:00 2001 From: Alexandru Gologan Date: Fri, 15 Mar 2024 06:36:53 +0200 Subject: [PATCH] Add support for a bookmarks-only mode (#716) * Add support for bookmarks-only mode * Add error for missing bookmarks in bookmarks-only mode * Error when settings url or connect backend together with bookmarks-only * Add tests for parsing options --- pkg/api/api.go | 9 ++++++--- pkg/command/options.go | 17 +++++++++++++++++ pkg/command/options_test.go | 14 ++++++++++++++ static/index.html | 4 +++- static/js/app.js | 24 +++++++++++++++++++++++- 5 files changed, 63 insertions(+), 5 deletions(-) diff --git a/pkg/api/api.go b/pkg/api/api.go index 05d293ab6..6d895817e 100644 --- a/pkg/api/api.go +++ b/pkg/api/api.go @@ -154,6 +154,8 @@ func Connect(c *gin.Context) { if bookmarkID := c.Request.FormValue("bookmark_id"); bookmarkID != "" { cl, err = ConnectWithBookmark(bookmarkID) + } else if command.Opts.BookmarksOnly { + err = errNotPermitted } else { cl, err = ConnectWithURL(c) } @@ -558,9 +560,10 @@ func GetInfo(c *gin.Context) { successResponse(c, gin.H{ "app": command.Info, "features": gin.H{ - "session_lock": command.Opts.LockSession, - "query_timeout": command.Opts.QueryTimeout, - "local_queries": QueryStore != nil, + "session_lock": command.Opts.LockSession, + "query_timeout": command.Opts.QueryTimeout, + "local_queries": QueryStore != nil, + "bookmarks_only": command.Opts.BookmarksOnly, }, }) } diff --git a/pkg/command/options.go b/pkg/command/options.go index 0e9fce23d..4e17b01e5 100644 --- a/pkg/command/options.go +++ b/pkg/command/options.go @@ -50,6 +50,7 @@ type Options struct { LockSession bool `long:"lock-session" description:"Lock session to a single database connection"` Bookmark string `short:"b" long:"bookmark" description:"Bookmark to use for connection. Bookmark files are stored under $HOME/.pgweb/bookmarks/*.toml" default:""` BookmarksDir string `long:"bookmarks-dir" description:"Overrides default directory for bookmark files to search" default:""` + BookmarksOnly bool `long:"bookmarks-only" description:"Allow only connections from bookmarks"` QueriesDir string `long:"queries-dir" description:"Overrides default directory for local queries"` DisablePrettyJSON bool `long:"no-pretty-json" description:"Disable JSON formatting feature for result export"` DisableSSH bool `long:"no-ssh" description:"Disable database connections via SSH"` @@ -118,6 +119,10 @@ func ParseOptions(args []string) (Options, error) { } } + if getPrefixedEnvVar("BOOKMARKS_ONLY") != "" { + opts.BookmarksOnly = true + } + if getPrefixedEnvVar("SESSIONS") != "" { opts.Sessions = true } @@ -162,6 +167,18 @@ func ParseOptions(args []string) (Options, error) { } } + if opts.BookmarksOnly { + if opts.URL != "" { + return opts, errors.New("--url not supported in bookmarks-only mode") + } + if opts.Host != "" && opts.Host != "localhost" { + return opts, errors.New("--host not supported in bookmarks-only mode") + } + if opts.ConnectBackend != "" { + return opts, errors.New("--connect-backend not supported in bookmarks-only mode") + } + } + homePath, err := homedir.Dir() if err != nil { fmt.Fprintf(os.Stderr, "[WARN] cant detect home dir: %v", err) diff --git a/pkg/command/options_test.go b/pkg/command/options_test.go index 565002d12..07b866fbe 100644 --- a/pkg/command/options_test.go +++ b/pkg/command/options_test.go @@ -80,4 +80,18 @@ func TestParseOptions(t *testing.T) { assert.NoError(t, err) assert.Equal(t, "../../data/passfile", opts.Passfile) }) + + t.Run("bookmarks only mode", func(t *testing.T) { + _, err := ParseOptions([]string{"--bookmarks-only"}) + assert.NoError(t, err) + + _, err = ParseOptions([]string{"--bookmarks-only", "--url", "test"}) + assert.EqualError(t, err, "--url not supported in bookmarks-only mode") + + _, err = ParseOptions([]string{"--bookmarks-only", "--host", "test", "--port", "5432"}) + assert.EqualError(t, err, "--host not supported in bookmarks-only mode") + + _, err = ParseOptions([]string{"--bookmarks-only", "--connect-backend", "test", "--sessions", "--connect-token", "token", "--url", "127.0.0.2"}) + assert.EqualError(t, err, "--connect-backend not supported in bookmarks-only mode") + }) } diff --git a/static/index.html b/static/index.html index 40b9bf2b8..3f1a57339 100644 --- a/static/index.html +++ b/static/index.html @@ -189,14 +189,16 @@

pgweb

-
+
+
+
diff --git a/static/js/app.js b/static/js/app.js index e3a2b6ab4..7347c5568 100644 --- a/static/js/app.js +++ b/static/js/app.js @@ -1091,6 +1091,7 @@ function showConnectionSettings() { // Show the current postgres version $(".connection-settings .version").text("v" + appInfo.version).show(); $("#connection_window").show(); + initConnectionWindow(); // Check github release page for updates getLatestReleaseInfo(appInfo); @@ -1119,11 +1120,32 @@ function showConnectionSettings() { $(".bookmarks").show(); } else { - $(".bookmarks").hide(); + if (appFeatures.bookmarks_only) { + $("#connection_error").html("Running in bookmarks-only mode but NO bookmarks configured.").show(); + $(".open-connection").hide(); + } else { + $(".bookmarks").hide(); + } } }); } +function initConnectionWindow() { + if (appFeatures.bookmarks_only) { + $(".connection-group-switch").hide(); + $(".connection-scheme-group").hide(); + $(".connection-bookmarks-group").show(); + $(".connection-standard-group").hide(); + $(".connection-ssh-group").hide(); + } else { + $(".connection-group-switch").show(); + $(".connection-scheme-group").hide(); + $(".connection-bookmarks-group").show(); + $(".connection-standard-group").show(); + $(".connection-ssh-group").hide(); + } +} + function getConnectionString() { var url = $.trim($("#connection_url").val()); var mode = $(".connection-group-switch button.active").attr("data");