Skip to content

Commit

Permalink
Add more EnumNamingStrategies (#4728)
Browse files Browse the repository at this point in the history
  • Loading branch information
lbenedetto authored Oct 2, 2024
1 parent 629af99 commit 26bdfec
Show file tree
Hide file tree
Showing 8 changed files with 706 additions and 269 deletions.
5 changes: 5 additions & 0 deletions release-notes/CREDITS-2.x
Original file line number Diff line number Diff line change
Expand Up @@ -1837,3 +1837,8 @@ Rikkarth (rikkarth@github)
Maxim Valeev (@MaximValeev)
* Reported #4508: Deserialized JsonAnySetter field in Kotlin data class is null
(2.18.1)


Lars Benedetto (@lbenedetto)
* Contributed #4676: Support other enum naming strategies than camelCase
(2.19.0)
4 changes: 3 additions & 1 deletion release-notes/VERSION-2.x
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,9 @@ Project: jackson-databind

2.19.0 (not yet released)

-
#4676: Support other enum naming strategies than camelCase
(requested by @hajdamak)
(contributed by Lars B)

2.18.1 (WIP-2024)

Expand Down
463 changes: 405 additions & 58 deletions src/main/java/com/fasterxml/jackson/databind/EnumNamingStrategies.java

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
import com.fasterxml.jackson.databind.introspect.AnnotatedField;
import com.fasterxml.jackson.databind.introspect.AnnotatedMethod;
import com.fasterxml.jackson.databind.introspect.AnnotatedParameter;
import com.fasterxml.jackson.databind.util.NamingStrategyImpls;

/**
* Container for standard {@link PropertyNamingStrategy} implementations
Expand Down Expand Up @@ -124,42 +125,6 @@ public String nameForConstructorParameter(MapperConfig<?> config, AnnotatedParam
}

public abstract String translate(String propertyName);

/**
* Helper method to share implementation between snake and dotted case.
*/
protected String translateLowerCaseWithSeparator(final String input, final char separator)
{
if (input == null || input.isEmpty()) {
return input;
}

final int length = input.length();
final StringBuilder result = new StringBuilder(length + (length >> 1));
int upperCount = 0;
for (int i = 0; i < length; ++i) {
char ch = input.charAt(i);
char lc = Character.toLowerCase(ch);

if (lc == ch) { // lower-case letter means we can get new word
// but need to check for multi-letter upper-case (acronym), where assumption
// is that the last upper-case char is start of a new word
if (upperCount > 1) {
// so insert hyphen before the last character now
result.insert(result.length() - 1, separator);
}
upperCount = 0;
} else {
// Otherwise starts new word, unless beginning of string
if ((upperCount == 0) && (i > 0)) {
result.append(separator);
}
++upperCount;
}
result.append(lc);
}
return result.toString();
}
}

/*
Expand Down Expand Up @@ -230,35 +195,7 @@ public static class SnakeCaseStrategy extends NamingBase
@Override
public String translate(String input)
{
if (input == null) return input; // garbage in, garbage out
int length = input.length();
StringBuilder result = new StringBuilder(length * 2);
int resultLength = 0;
boolean wasPrevTranslated = false;
for (int i = 0; i < length; i++)
{
char c = input.charAt(i);
if (i > 0 || c != '_') // skip first starting underscore
{
if (Character.isUpperCase(c))
{
if (!wasPrevTranslated && resultLength > 0 && result.charAt(resultLength - 1) != '_')
{
result.append('_');
resultLength++;
}
c = Character.toLowerCase(c);
wasPrevTranslated = true;
}
else
{
wasPrevTranslated = false;
}
result.append(c);
resultLength++;
}
}
return resultLength > 0 ? result.toString() : input;
return NamingStrategyImpls.SNAKE_CASE.translate(input);
}
}

Expand All @@ -281,11 +218,7 @@ public static class UpperSnakeCaseStrategy extends SnakeCaseStrategy

@Override
public String translate(String input) {
String output = super.translate(input);
if (output == null) {
return null;
}
return output.toUpperCase();
return NamingStrategyImpls.UPPER_SNAKE_CASE.translate(input);
}
}

Expand All @@ -305,7 +238,7 @@ public static class LowerCamelCaseStrategy extends NamingBase

@Override
public String translate(String input) {
return input;
return NamingStrategyImpls.LOWER_CAMEL_CASE.translate(input);
}
}

Expand Down Expand Up @@ -343,18 +276,7 @@ public static class UpperCamelCaseStrategy extends NamingBase
*/
@Override
public String translate(String input) {
if (input == null || input.isEmpty()){
return input; // garbage in, garbage out
}
// Replace first lower-case letter with upper-case equivalent
char c = input.charAt(0);
char uc = Character.toUpperCase(c);
if (c == uc) {
return input;
}
StringBuilder sb = new StringBuilder(input);
sb.setCharAt(0, uc);
return sb.toString();
return NamingStrategyImpls.UPPER_CAMEL_CASE.translate(input);
}
}

Expand All @@ -376,10 +298,7 @@ public static class LowerCaseStrategy extends NamingBase

@Override
public String translate(String input) {
if (input == null || input.isEmpty()) {
return input;
}
return input.toLowerCase();
return NamingStrategyImpls.LOWER_CASE.translate(input);
}
}

Expand All @@ -401,7 +320,7 @@ public static class KebabCaseStrategy extends NamingBase

@Override
public String translate(String input) {
return translateLowerCaseWithSeparator(input, '-');
return NamingStrategyImpls.KEBAB_CASE.translate(input);
}
}

Expand All @@ -422,7 +341,7 @@ public static class LowerDotCaseStrategy extends NamingBase

@Override
public String translate(String input){
return translateLowerCaseWithSeparator(input, '.');
return NamingStrategyImpls.LOWER_DOT_CASE.translate(input);
}
}
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,157 @@
package com.fasterxml.jackson.databind.util;

/**
* Container for standard naming strategy implementations, specifically
* used by property naming strategies (see {@link com.fasterxml.jackson.databind.PropertyNamingStrategies})
* and enum naming strategies (see {@link com.fasterxml.jackson.databind.EnumNamingStrategies}).
*/
public enum NamingStrategyImpls {
/**
* beanName -> beanName
*/
LOWER_CAMEL_CASE {
@Override
public String translate(String beanName) {
return beanName; // beanName is already in lower camel case
}
},

/**
* beanName -> BeanName
*/
UPPER_CAMEL_CASE {
@Override
public String translate(String beanName) {
if (beanName == null || beanName.isEmpty()) {
return beanName; // garbage in, garbage out
}
// Replace first lower-case letter with upper-case equivalent
char c = beanName.charAt(0);
char uc = Character.toUpperCase(c);
if (c == uc) {
return beanName;
}
StringBuilder sb = new StringBuilder(beanName);
sb.setCharAt(0, uc);
return sb.toString();
}
},

/**
* beanName -> bean_name
*/
SNAKE_CASE {
@Override
public String translate(String beanName) {
if (beanName == null) return beanName; // garbage in, garbage out
int length = beanName.length();
StringBuilder result = new StringBuilder(length * 2);
int resultLength = 0;
boolean wasPrevTranslated = false;
for (int i = 0; i < length; i++) {
char c = beanName.charAt(i);
if (i > 0 || c != '_') // skip first starting underscore
{
if (Character.isUpperCase(c)) {
if (!wasPrevTranslated && resultLength > 0 && result.charAt(resultLength - 1) != '_') {
result.append('_');
resultLength++;
}
c = Character.toLowerCase(c);
wasPrevTranslated = true;
} else {
wasPrevTranslated = false;
}
result.append(c);
resultLength++;
}
}
return resultLength > 0 ? result.toString() : beanName;
}
},

/**
* beanName -> BEAN_NAME
*/
UPPER_SNAKE_CASE {
@Override
public String translate(String beanName) {
String output = SNAKE_CASE.translate(beanName);
if (output == null) {
return null;
}
return output.toUpperCase();
}
},

/**
* beanName -> beanname
*/
LOWER_CASE {
@Override
public String translate(String beanName) {
if (beanName == null || beanName.isEmpty()) {
return beanName;
}
return beanName.toLowerCase();
}
},

/**
* beanName -> bean-name
*/
KEBAB_CASE {
@Override
public String translate(String beanName) {
return translateLowerCaseWithSeparator(beanName, '-');
}
},

/**
* beanName -> bean.name
*/
LOWER_DOT_CASE {
@Override
public String translate(String beanName) {
return translateLowerCaseWithSeparator(beanName, '.');
}
},
;

public abstract String translate(final String beanName);

/**
* Helper method to share implementation between snake and dotted case.
*/
static String translateLowerCaseWithSeparator(final String beanName, final char separator) {
if (beanName == null || beanName.isEmpty()) {
return beanName;
}

final int length = beanName.length();
final StringBuilder result = new StringBuilder(length + (length >> 1));
int upperCount = 0;
for (int i = 0; i < length; ++i) {
char ch = beanName.charAt(i);
char lc = Character.toLowerCase(ch);

if (lc == ch) { // lower-case letter means we can get new word
// but need to check for multi-letter upper-case (acronym), where assumption
// is that the last upper-case char is start of a new word
if (upperCount > 1) {
// so insert hyphen before the last character now
result.insert(result.length() - 1, separator);
}
upperCount = 0;
} else {
// Otherwise starts new word, unless beginning of string
if ((upperCount == 0) && (i > 0)) {
result.append(separator);
}
++upperCount;
}
result.append(lc);
}
return result.toString();
}
}
Loading

0 comments on commit 26bdfec

Please sign in to comment.