Skip to content
This repository has been archived by the owner on Oct 15, 2024. It is now read-only.

rename updates #593

Merged
merged 7 commits into from
Apr 24, 2016
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
39 changes: 38 additions & 1 deletion src/plugins/rename/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,14 +8,51 @@
- infos/metadata = rename/to rename/toupper rename/tolower rename/cut
- infos/description = renaming of keys

## INTRODUCTION ##
# INTRODUCTION #


This plugin can be used to perform rename operations on keys. This is useful if a backend does not provide keys
with the required names or case.

If keys are renamed, their original name is stored in the `origname` MetaKey.

There are 2 types of transformations:
* basic
* advanced


# BASIC TRANSFORMATIONS #

are applied before and after the advanced transformations.

## GET ##

first transformation to be applied to all keys on kdbGet.

`get/case`
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

one sentence here would be good.

* toupper
* tolower
* unchanged // this is the default value if no configuration is given

converts the whole keyname below the parentKey to upper- or lowercase. if no configuration or `unchanged` is used no transformation is done here.

## SET ##

last transformation to be applied to all keys on kdbSet.

`set/case`
* toupper
* tolower
* keyname
* unchanged // this is the default value if no configuration is given

`toupper` or `tolower` tells the rename plugin to convert the whole keyname below below the parentKey to lower or uppercase.

`unchanged` returnes the key to it's original name

`keyname` tells the plugin to keep the name of the advanced transformation
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

one example would be great!


# ADVANCED TRANSFORMATIONS #

## CUT ##

Expand Down
114 changes: 95 additions & 19 deletions src/plugins/rename/rename.c
Original file line number Diff line number Diff line change
Expand Up @@ -26,10 +26,12 @@
#define ELEKTRA_ORIGINAL_NAME_META "origname"
#define TOLOWER (-1)
#define TOUPPER 1
#define UNCHNGD 0
#define KEYNAME 2
// TODO defined privately in keyhelpers.c, API break possible..
char * keyNameGetOneLevel (const char * name, size_t * size);

static void doConversion (char * newName, int levels, int toCase)
static void doConversion (char * newName, int levels, const int toCase)
{
int (*conversion) (int);

Expand Down Expand Up @@ -74,7 +76,7 @@ static void doConversion (char * newName, int levels, int toCase)
}

Key * elektraKeyCreateNewName (const Key * key, const Key * parentKey, const char * cutPath, const char * replaceWith,
const char * toUpperPath, const char * toLowerPath)
const char * toUpperPath, const char * toLowerPath, const int initialConversion)
{
size_t addToLen = 0;
if (replaceWith != NULL) addToLen = strlen (replaceWith);
Expand All @@ -89,9 +91,15 @@ Key * elektraKeyCreateNewName (const Key * key, const Key * parentKey, const cha
char * curKeyName = elektraMalloc (keyGetFullNameSize (key));
keyGetFullName (key, curKeyName, keyGetFullNameSize (key));

const char * afterParentString = curKeyName + (strlen (parentKeyName));
char * afterParentString = curKeyName + (strlen (parentKeyName));
char * ptr;

if (initialConversion != UNCHNGD)
{
doConversion (afterParentString, 0, initialConversion);
replace = 1;
}

if (cutPath && (cutPath[0] != '/') && ((ptr = strstr (afterParentString, cutPath)) != NULL))
{
strncpy (newName, afterParentString, (ptr - afterParentString));
Expand Down Expand Up @@ -165,7 +173,8 @@ static void keyAddUnescapedBasePath (Key * key, const char * path)
}
}

static Key * renameGet (Key * key, Key * parentKey, Key * cutConfig, Key * replaceWithConfig, Key * toUpperConfig, Key * toLowerConfig)
static Key * renameGet (Key * key, Key * parentKey, Key * cutConfig, Key * replaceWithConfig, Key * toUpperConfig, Key * toLowerConfig,
Key * getCase)
{
char * cutPath = 0;
char * replaceWith = 0;
Expand All @@ -175,6 +184,24 @@ static Key * renameGet (Key * key, Key * parentKey, Key * cutConfig, Key * repla
const Key * toMeta = keyGetMeta (key, "rename/to");
const Key * toUpperMeta = keyGetMeta (key, "rename/toupper");
const Key * toLowerMeta = keyGetMeta (key, "rename/tolower");

int initialConversion = 0;
if (getCase)
{
const char * str = keyString (getCase);
if (!strcmp (str, "toupper"))
{
initialConversion = TOUPPER;
}
else if (!strcmp (str, "tolower"))
{
initialConversion = TOLOWER;
}
else
{
initialConversion = UNCHNGD;
}
}
/* if the meta config exists, it takes precedence over the global config */
if (cutMeta)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why did you remove these lines which restore the sync bit? They are needed for #404. We should always put comments above lines that fix a bug so that everyone know what they are for.

cutPath = (char *)keyString (cutMeta);
Expand All @@ -193,7 +220,7 @@ static Key * renameGet (Key * key, Key * parentKey, Key * cutConfig, Key * repla
else if (toLowerConfig)
toLowerPath = (char *)keyString (toLowerConfig);

return elektraKeyCreateNewName (key, parentKey, cutPath, replaceWith, toUpperPath, toLowerPath);
return elektraKeyCreateNewName (key, parentKey, cutPath, replaceWith, toUpperPath, toLowerPath, initialConversion);
}

static Key * restoreKeyName (Key * key, const Key * parentKey, const Key * configKey)
Expand All @@ -207,11 +234,6 @@ static Key * restoreKeyName (Key * key, const Key * parentKey, const Key * confi
Key * result = keyDup (key);
keySetName (result, keyString (origNameKey));
keySetMeta (result, ELEKTRA_ORIGINAL_NAME_META, 0);

if (!hasSync)
{
keyClearSync (result);
}
return result;
}
}
Expand Down Expand Up @@ -265,10 +287,14 @@ int elektraRenameGet (Plugin * handle, KeySet * returned, Key * parentKey)
Key * toUpper = ksLookupByName (config, "/toupper", KDB_O_NONE);
Key * toLower = ksLookupByName (config, "/tolower", KDB_O_NONE);
Key * replaceWith = ksLookupByName (config, "/replacewith", KDB_O_NONE);
Key * getCase = ksLookupByName (config, "/get/case", KDB_O_NONE);


Key * key;
while ((key = ksNext (iterateKs)) != 0)
{
Key * renamedKey = renameGet (key, parentKey, cutConfig, replaceWith, toUpper, toLower);

Key * renamedKey = renameGet (key, parentKey, cutConfig, replaceWith, toUpper, toLower, getCase);

if (renamedKey)
{
Expand Down Expand Up @@ -315,29 +341,80 @@ int elektraRenameSet (Plugin * handle, KeySet * returned, Key * parentKey)
KeySet * config = elektraPluginGetConfig (handle);
Key * cutConfig = ksLookupByName (config, "/cut", KDB_O_NONE);

Key * setCase = ksLookupByName (config, "/set/case", KDB_O_NONE);

int writeConversion = 0;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think its nicer when you leave this customized formatted, don't you agree?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i honestly have no idea why that line always disappears...

if (setCase)
{
const char * str = keyString (setCase);
if (!strcmp (str, "toupper"))
{
writeConversion = TOUPPER;
}
else if (!strcmp (str, "tolower"))
{
writeConversion = TOLOWER;
}
else if (!strcmp (str, "keyname"))
{
writeConversion = KEYNAME;
}
else
{
writeConversion = UNCHNGD;
}
}
ksRewind (iterateKs);
Key * key;
char * parentKeyName = elektraMalloc (keyGetFullNameSize (parentKey));
keyGetFullName (parentKey, parentKeyName, keyGetFullNameSize (parentKey));
while ((key = ksNext (iterateKs)) != 0)
{
Key * renamedKey = restoreKeyName (key, parentKey, cutConfig);

if (renamedKey)
Key * renamedKey = NULL;
if (writeConversion != KEYNAME)
{
renamedKey = restoreKeyName (key, parentKey, cutConfig);

if (!renamedKey) renamedKey = keyDup (key);
if (writeConversion == TOUPPER || writeConversion == TOLOWER)
{
char * curKeyName = elektraMalloc (keyGetFullNameSize (renamedKey));
keyGetFullName (renamedKey, curKeyName, keyGetFullNameSize (renamedKey));

char * afterParentString = curKeyName + (strlen (parentKeyName));

doConversion (afterParentString, 0, writeConversion);

keySetName (renamedKey, curKeyName);
elektraFree (curKeyName);
}
/*
* if something is restored from the parentKey, do
* not delete the parentKey (might cause troubles)
*/
if (keyCmp (key, parentKey) != 0)
{
ksLookup (returned, key, KDB_O_POP);
keyDel (key);
keyDel (ksLookup (returned, key, KDB_O_POP));
}
ksAppendKey (returned, renamedKey);
keyDel (renamedKey);
}
else
{
if (keyCmp (key, parentKey) != 0)
{
keyDel (ksLookupByName (returned, keyString (keyGetMeta (key, ELEKTRA_ORIGINAL_NAME_META)), KDB_O_POP));
}
ksAppendKey (returned, key);
}
}


keyIncRef (parentKey);
ksDel (iterateKs);

keyDecRef (parentKey);

ksRewind (returned);
elektraFree (parentKeyName);
return 1; /* success */
}

Expand All @@ -349,4 +426,3 @@ Plugin * ELEKTRA_PLUGIN_EXPORT (rename)
ELEKTRA_PLUGIN_SET, &elektraRenameSet,
ELEKTRA_PLUGIN_END);
}

2 changes: 1 addition & 1 deletion src/plugins/rename/rename.h
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@
int elektraRenameGet (Plugin * handle, KeySet * ks, Key * parentKey);
int elektraRenameSet (Plugin * handle, KeySet * ks, Key * parentKey);
Key * elektraKeyCreateNewName (const Key * key, const Key * parentKey, const char * cutPath, const char * replaceWith, const char * toUpper,
const char * toLower);
const char * toLower, const int initialConversion);

Plugin * ELEKTRA_PLUGIN_EXPORT (rename);

Expand Down
30 changes: 25 additions & 5 deletions src/plugins/rename/testmod_rename.c
Original file line number Diff line number Diff line change
Expand Up @@ -225,28 +225,28 @@ static void test_metaConfigTakesPrecedence ()
static void test_keyCutNamePart ()
{
Key * parentKey = keyNew ("user/tests/rename", KEY_END);
Key * result = elektraKeyCreateNewName (parentKey, parentKey, "wont/cut/this", NULL, NULL, NULL);
Key * result = elektraKeyCreateNewName (parentKey, parentKey, "wont/cut/this", NULL, NULL, NULL, 0);
succeed_if (!result, "parentKey was modified although it should have been ignored");

/* cutting works correctly without trailing slash */
Key * testKey = keyNew ("user/tests/rename/will/cut/this/key1", KEY_END);
result = elektraKeyCreateNewName (testKey, parentKey, "will/cut/this", NULL, NULL, NULL);
result = elektraKeyCreateNewName (testKey, parentKey, "will/cut/this", NULL, NULL, NULL, 0);
succeed_if (result, "key1 was not cut")
succeed_if (!strcmp (keyName (result), "user/tests/rename/key1"), "cutting key1 did not yield the expected result");
keyDel (testKey);
keyDel (result);

/* cutting works correctly with trailing slash */
testKey = keyNew ("user/tests/rename/will/cut/this/key1", KEY_END);
result = elektraKeyCreateNewName (testKey, parentKey, "will/cut/this/", NULL, NULL, NULL);
result = elektraKeyCreateNewName (testKey, parentKey, "will/cut/this/", NULL, NULL, NULL, 0);
succeed_if (result, "key1 was not cut")
succeed_if (!strcmp (keyName (result), "user/tests/rename/key1"), "cutting key1 did not yield the expected result");
keyDel (testKey);
keyDel (result);

/* disallow leading slashes */
testKey = keyNew ("user/tests/rename/wont/cut/this/key1", KEY_END);
result = elektraKeyCreateNewName (testKey, parentKey, "/wont/cut/this", NULL, NULL, NULL);
result = elektraKeyCreateNewName (testKey, parentKey, "/wont/cut/this", NULL, NULL, NULL, 0);
succeed_if (!result, "key was cut although it the cutpath contained a leading slash");
keyDel (testKey);
keyDel (parentKey);
Expand Down Expand Up @@ -395,6 +395,26 @@ static void test_mixCase ()
PLUGIN_CLOSE ();
}

static void test_write ()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can you also test the other direction (upper/lower)

{
Key * parentKey = keyNew ("user/tests/rename", KEY_END);
KeySet * conf =
ksNew (20, keyNew ("system/tolower", KEY_VALUE, "1", KEY_END), keyNew ("system/get/case", KEY_VALUE, "toupper", KEY_END),
keyNew ("system/set/case", KEY_VALUE, "keyname"), KS_END);
KeySet * ks = ksNew (20, keyNew ("user/tests/rename/uppercase/uppercase/uppercase/LOWERCASE", KEY_VALUE, "test", KEY_END), KS_END);
ksAppendKey (ks, parentKey);
PLUGIN_OPEN ("rename");
succeed_if (plugin->kdbGet (plugin, ks, parentKey) >= 1, "call to kdbGet was not successful");
Key * key = ksLookupByName (ks, "user/tests/rename/UPPERCASE/UPPERCASE/UPPERCASE/lowercase", KDB_O_NONE);
succeed_if (key, "key1 was not correctly rename");
succeed_if (plugin->kdbSet (plugin, ks, parentKey) == 1, "call to kdbSet was not successful");
key = ksLookupByName (ks, "user/tests/rename/UPPERCASE/UPPERCASE/UPPERCASE/lowercase", KDB_O_NONE);
succeed_if (key, "key1s name was not correctly saved");
keyDel (parentKey);
ksDel (ks);
PLUGIN_CLOSE ();
}

int main (int argc, char ** argv)
{
printf ("RENAME TESTS\n");
Expand All @@ -415,7 +435,7 @@ int main (int argc, char ** argv)
test_toLower ();
test_mixCase ();
test_replaceString ();

test_write ();
printf ("\ntest_rename RESULTS: %d test(s) done. %d error(s).\n", nbTest, nbError);

return nbError;
Expand Down