Skip to content

Commit

Permalink
kdb-cli: rewrite set
Browse files Browse the repository at this point in the history
... and (partial) fix #3742, fix #4028, fix #1164
  • Loading branch information
hannes99 committed May 19, 2023
1 parent 57ba9a7 commit 8d4295e
Show file tree
Hide file tree
Showing 11 changed files with 163 additions and 171 deletions.
34 changes: 4 additions & 30 deletions doc/tutorials/cascading.md
Original file line number Diff line number Diff line change
Expand Up @@ -91,39 +91,13 @@ The **proc** namespace is not accessible by the command line tool **kdb**, as it

The **spec** namespace is used to store metadata about keys and therefore Elektra handles the **spec** namespace differently to other namespaces. The following part of the tutorial is dedicated to the impact of the **spec** namespace on cascading lookups.

## Write Operations and the cascading Namespace
## Cascading writes are not possible

When performing writing operations, a namespace always has to be specified.

```sh
kdb set /tests/tutorial/cascading/#0/current/cascading_write_test value
# STDERR: Aborting: A cascading write to a non-existent key is ambiguous.
# RET: 12

kdb meta-set /tests/tutorial/cascading/#0/current/cascading_write_test metakey metavalue
# STDERR: Aborting: A cascading write to a non-existent key is ambiguous.
# RET: 12
```

will both fail, as no matching key exists.
Since there are multiple hypothetical key names that would match the cascading name (keys of specific namespaces like user:, system:, ...) if they existed, it is not clear what the user intended and thus an error is produced.

To make the previous two operations meaningful, a matching key in the user: namespace is created:

```sh
kdb set user:/tests/tutorial/cascading/#0/current/cascading_write_test value
#> Create a new key user:/tests/tutorial/cascading/#0/current/cascading_write_test with string "value"
```

Now, the operations operate on a well-defined key and succeed this time:

```sh
kdb set /tests/tutorial/cascading/#0/current/cascading_write_test value
#> Set string to "value"
#> Using name user:/tests/tutorial/cascading/#0/current/cascading_write_test

kdb meta-set /tests/tutorial/cascading/#0/current/cascading_write_test metakey metavalue
#> Using name user:/tests/tutorial/cascading/#0/current/cascading_write_test
kdb set /tests/tutorial/cascading/key1 "hello world"
# RET: 2
# STDERR: .*key does not specify a namespace
```

## Override Links
Expand Down
2 changes: 1 addition & 1 deletion src/plugins/email/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ The package is called `libelektra5-experimental`.
kdb mount config.dump /tests/email dump email

# Incorrect address is valid with incomplete configuration
kdb meta-set spec:/tests/email/noaddr check/email
# kdb meta set spec:/tests/email/noaddr check/email
kdb set user:/tests/email/noaddr invalid..address@com
# RET: 0

Expand Down
4 changes: 2 additions & 2 deletions src/plugins/hosts/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -98,10 +98,10 @@ kdb get /tests/hosts/ipv6/localhost
#> ::1

# Should both fail with error C03200 and return 5
kdb set /tests/hosts/ipv4/localhost ::1
kdb set user:/tests/hosts/ipv4/localhost ::1
# RET:5
# ERROR:C03200
kdb set /tests/hosts/ipv6/localhost 127.0.0.1
kdb set user:/tests/hosts/ipv6/localhost 127.0.0.1
# RET:5
# ERROR:C03200

Expand Down
2 changes: 1 addition & 1 deletion src/plugins/kconfig/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,7 @@ kdb get /tests/kconfig/key
#> Value

# Set the value to Example
kdb set /tests/kconfig/key Example
kdb set user:/tests/kconfig/key Example

# Verify that the value has changed in the file too
cat `kdb file user:/tests/kconfig`
Expand Down
1 change: 1 addition & 0 deletions src/plugins/range/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@ kdb set user:/tests/range/value 2
# RET:0

kdb rm -r user:/tests/range
kdb rm -r spec:/tests/range
sudo kdb umount /tests/range
```

Expand Down
2 changes: 0 additions & 2 deletions src/tools/kdb/factory.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,6 @@
#include <pluginlist.hpp>
#include <remount.hpp>
#include <rm.hpp>
#include <set.hpp>
#include <sget.hpp>
#include <shell.hpp>
#include <showmeta.hpp>
Expand Down Expand Up @@ -81,7 +80,6 @@ class Factory
Factory () : m_factory ()
{
// TODO: to add a new command, 2.) add a line here -> and you are done
m_factory.insert (std::make_pair ("set", std::make_shared<Cnstancer<SetCommand>> ()));
m_factory.insert (std::make_pair ("rm", std::make_shared<Cnstancer<RemoveCommand>> ()));
m_factory.insert (std::make_pair ("cache", std::make_shared<Cnstancer<CacheCommand>> ()));
m_factory.insert (std::make_pair ("complete", std::make_shared<Cnstancer<CompleteCommand>> ()));
Expand Down
2 changes: 2 additions & 0 deletions src/tools/kdb/main.c
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
#include <ls.h>
#include <merge.h>
#include <namespace.h>
#include <set.h>

#include <command.h>
#include <kdb.h>
Expand All @@ -33,6 +34,7 @@ command subcommands[] = {
{ "get", addGetSpec, execGet },
{ "ls", addLsSpec, execLs },
{ "namespace", addNamespaceSpec, execNamespace },
{ "set", addSetSpec, execSet },
};

void printError (Key * errorKey)
Expand Down
119 changes: 119 additions & 0 deletions src/tools/kdb/set.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
/**
* @file
*
* @brief KDB set subcommand
*
* @copyright BSD License (see LICENSE.md or https://www.libelektra.org)
*/

#include <colors.h>
#include <command.h>
#include <kdbassert.h>
#include <kdbease.h>
#include <kdberrors.h>
#include <set.h>
#include <stdio.h>
#include <string.h>

#define COMMAND_NAME "set"

#define GET_OPTION_KEY(options, name) GET_OPT_KEY (options, COMMAND_BASE_KEY (COMMAND_NAME) "/" name)
#define GET_OPTION(options, name) GET_OPT (options, COMMAND_BASE_KEY (COMMAND_NAME) "/" name)

void addSetSpec (KeySet * spec)
{
ksAppendKey (spec, keyNew (COMMAND_SPEC_KEY (COMMAND_NAME), KEY_META, "description", "Set the value of an individual key.",
KEY_META, "command", COMMAND_NAME, KEY_END));
ksAppendKey (spec, keyNew (COMMAND_SPEC_KEY (COMMAND_NAME) "/force", KEY_META, "description", "Force setting the value", KEY_META,
"opt", "f", KEY_META, "opt/long", "force", KEY_META, "opt/arg", "none", KEY_END));
ksAppendKey (spec, keyNew (COMMAND_SPEC_KEY (COMMAND_NAME) "/name", KEY_META, "description", "The name of the key", KEY_META,
"args", "indexed", KEY_META, "args/index", "0", KEY_END));

ksAppendKey (spec, keyNew (COMMAND_SPEC_KEY (COMMAND_NAME) "/value", KEY_META, "description", "The value that should be set",
KEY_META, "args", "indexed", KEY_META, "args/index", "1", KEY_END));

ADD_BASIC_OPTIONS (spec, COMMAND_SPEC_KEY (COMMAND_NAME))
}

int execSet (KeySet * options, Key * errorKey)
{
int ret = 0;
GET_BASIC_OPTIONS

bool force = false;
tmp = GET_OPTION_KEY (options, "force");
if (tmp != NULL)
{
elektraKeyToBoolean (GET_OPTION_KEY (options, "force"), &force);
}

const char * name = getKeyNameFromOptions (GET_OPTION (options, "name"), errorKey, verbose);
if (name == NULL)
{
RETURN (2)
}

const char * value = GET_OPTION (options, "value");

Key * parentKey = keyNew (name, KEY_END);

if (keyGetNamespace (parentKey) == KEY_NS_NONE || keyGetNamespace (parentKey) == KEY_NS_CASCADING)
{
ELEKTRA_SET_VALIDATION_SYNTACTIC_ERROR (errorKey, "key does not specify a namespace");
elektraFree ((void *) name);
keyDel (parentKey);
RETURN (2)
}

Key * maybeCascadingParent = keyDup (parentKey, KEY_CP_NAME);
if (!force)
{
keySetNamespace (maybeCascadingParent, KEY_NS_CASCADING);
}
KeySet * conf = ksNew (0, KS_END);
KDB * handle = kdbOpen (NULL, errorKey);

if (kdbGet (handle, conf, maybeCascadingParent) == -1)
{
ELEKTRA_SET_VALIDATION_SEMANTIC_ERRORF (errorKey, "could not load '%s': %s", keyName (parentKey),
GET_ERR (maybeCascadingParent));
ret = 5;
goto cleanup;
}
keyCopyAllMeta (errorKey, parentKey);

Key * key = ksLookup (conf, parentKey, KDB_O_NONE);
if (key == NULL)
{
key = keyNew (name, KEY_END);
ksAppendKey (conf, key);
CLI_PRINT (CLI_LOG_NONE, "Create a new key %s with string \"%s\"", keyName (key), value);
}
else
{
CLI_PRINT (CLI_LOG_NONE, "Set string to \"%s\"", value);
}
keySetString (key, value); // can't fail, since neither value or key can be null

if (kdbSet (handle, conf, parentKey) == -1)
{
ret = 5;
ELEKTRA_SET_VALIDATION_SEMANTIC_ERRORF (errorKey, "could not set value for '%s': %s", name, GET_ERR (parentKey));
}
keyCopyAllMeta (errorKey, parentKey);

keyDel (key);

cleanup:
if (!noNewLine)
{
printf ("\n");
}
elektraFree ((void *) name);
keyDel (parentKey);
keyDel (maybeCascadingParent);
ksDel (conf);
kdbClose (handle, errorKey);

RETURN (ret)
}
86 changes: 0 additions & 86 deletions src/tools/kdb/set.cpp

This file was deleted.

33 changes: 33 additions & 0 deletions src/tools/kdb/set.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
/**
* @file
*
* @brief KDB set subcommand header
*
* @copyright BSD License (see LICENSE.md or https://www.libelektra.org)
*/

#ifndef ELEKTRA_KDB_SET_H
#define ELEKTRA_KDB_SET_H

#include <kdb.h>

/**
* Adds options specification of set command to keySet
*
* @param spec the base spec where the commands spec should be added
*/
void addSetSpec (KeySet * spec);

/**
* Runs the set command
*
* @param options cli options and arguments as specified in addSetSpec()
* @param errorKey key where errors and warnings should be saved
*
* @retval 0 set command ran without errors
* @retval 1 errors occurred, keySetMeta (errorKey, "error/reason") for info
*
*/
int execSet (KeySet * options, Key * errorKey);

#endif // ELEKTRA_KDB_SET_H

0 comments on commit 8d4295e

Please sign in to comment.