Last active
February 20, 2019 04:42
-
-
Save SwingGuy1024/fd8ee13838cba4856913ffbd90f1944e to your computer and use it in GitHub Desktop.
Encoder/Decoder
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package com.neptunedreams.tools; | |
import com.google.gson.Gson; | |
import com.google.gson.GsonBuilder; | |
import com.google.gson.JsonObject; | |
import com.google.gson.JsonParser; | |
import java.io.ByteArrayOutputStream; | |
import java.nio.charset.Charset; | |
import java.util.HashSet; | |
import java.util.Set; | |
import java.util.StringTokenizer; | |
/** | |
* <p>Created by IntelliJ IDEA. | |
* <p>Date: 2/17/18 | |
* <p>Time: 6:26 PM | |
* | |
* @author Miguel Mu\u00f1oz | |
*/ | |
@SuppressWarnings({"WeakerAccess", "MagicCharacter", "MagicNumber", "HardcodedLineSeparator"}) | |
public final class UrlConvert { | |
private static Charset utf_8 = Charset.forName("UTF-8"); | |
private UrlConvert() { | |
} | |
public static String decodeAscii(final String input) { | |
ByteArrayOutputStream outputStream = new ByteArrayOutputStream(); | |
int tail = 2; | |
int ii = 0; | |
while (ii < (input.length() - tail)) { | |
char c = input.charAt(ii); | |
if ((c == '%') && isCode(input, ii + 1)) { | |
outputStream.write(getCode(input.substring(ii + 1, ii + 3))); | |
ii += 3; | |
} else if (c > 0xFF) { | |
int ic = (int) c; | |
throw new IllegalArgumentException(String.format("Character %c (%d = 0x%02X) too high", c, ic, ic)); | |
} else { | |
outputStream.write(c); | |
ii++; | |
} | |
} | |
while (ii < input.length()) { | |
outputStream.write(input.charAt(ii)); | |
++ii; | |
} | |
return toUtf8(outputStream.toByteArray()); | |
} | |
public static String encodeAscii(String input) { | |
StringBuilder builder = new StringBuilder(); | |
byte[] utf8Text = toByteData(input); | |
for (byte b : utf8Text) { | |
char c = (char) b; | |
if (charsToSkip.contains(c)) { | |
builder.append(c); | |
} else { | |
builder.append(String.format("%%%02X", b));// NON-NLS | |
} | |
} | |
return builder.toString(); | |
} | |
// public static void charTally(String s) { | |
// Set<Character> cSet = new TreeSet<>(); | |
// for (int ii=0; ii<s.length(); ++ii) { | |
// cSet.add(s.charAt(ii)); | |
// } | |
// for (Character c: cSet) { | |
// showChar(c); | |
// } | |
// System.out.println("\n\nMissing:"); | |
// for (char c=32; c<0x7f; ++c) { | |
// if (!cSet.contains(c)) { | |
// showChar(c); | |
// } | |
// } | |
// | |
// Set<Character> codedSet = new TreeSet<>(); | |
// String encoded = decodeAscii(s); | |
// for (int i=0; i<encoded.length(); ++i) { | |
// char c = encoded.charAt(i); | |
// if (!cSet.contains(c)) { | |
// codedSet.add(c); | |
// } | |
// } | |
// System.out.println("\n\nCoded:"); | |
// for (char c: codedSet) { | |
// System.out.print(c); | |
// } | |
// System.out.println(""); | |
// } | |
// private static void showChar(Character c) { | |
// System.out.printf("%c (%2d = 0x%02X)%n", c, (int)c, (int)c); | |
// } | |
private static int getCode(String s) { | |
return (char) Integer.parseUnsignedInt(s, 16); | |
} | |
private static boolean isCode(String s, int ii) { | |
return (isHex(s.charAt(ii)) && isHex(s.charAt(ii + 1))); | |
} | |
private static boolean isHex(char c) { | |
if ((c >= '0') && (c <= '9')) { | |
return true; | |
} | |
if ((c >= 'A') && (c <= 'F')) { | |
return true; | |
} | |
return (c >= 'a') && (c <= 'f'); | |
} | |
@SuppressWarnings("HardCodedStringLiteral") | |
public static void main(String[] args) { | |
if (args.length > 0) { | |
String start = args[0]; | |
String middle = decodeAscii(start); | |
String end = encodeAscii(middle); | |
if (start.equals(end)) { | |
System.out.println("Passed"); | |
} else { | |
System.out.println("Failed"); | |
System.out.printf("Lengths: %d & %d%n", start.length(), end.length()); | |
int j = 0; | |
for (int ii = 0; ii < start.length(); ++ii) { | |
if (start.charAt(ii) != end.charAt(j)) { | |
System.out.printf("Mismatch at %d: %s, %s%n", ii, start.substring(ii, ii + 10), end.substring(j, j + 10)); | |
j += 2; | |
} | |
j++; | |
} | |
} | |
} | |
// charTally(args[0]); | |
// test(); | |
} | |
// private static void test() { | |
// test("%7bBraces%7D", "{Braces}"); | |
// test("x%7bBraces%7dx", "x{Braces}x"); | |
// test("xy%7bBraces%7dxy", "xy{Braces}xy"); | |
// test("abc%3cdef%3Eghi", "abc<def>ghi"); | |
// test("%7BBraces%7", "{Braces%7"); | |
// test("%7BBraces%", "{Braces%"); | |
// test("Brace %7g %3C %3E", "Brace %7g < >"); | |
// test("Brace %g7 %3C %3E", "Brace %g7 < >"); | |
// testStrip("abc\n def\n ghi\njkl\n ", "abcdefghijkl"); | |
// testStrip("abc \n def\n ghi\njkl\n ", "abc defghijkl"); | |
// testStrip("abc \n de f\n ghi\njkl\nmno", "abc de fghijklmno"); | |
// } | |
public static String stripLineBreaks(String txt) { | |
int fromIndex = 0; | |
StringBuilder builder = new StringBuilder(); | |
int newLineSpot = txt.indexOf('\n'); | |
boolean trailingSpaceFound = false; | |
while (newLineSpot >= 0) { | |
builder.append(txt, fromIndex, newLineSpot); | |
int index = newLineSpot + 1; | |
char space = txt.charAt(index++); | |
trailingSpaceFound = false; | |
while (((space == ' ') || (space == '\n')) && (index < txt.length())) { | |
space = txt.charAt(index++); | |
trailingSpaceFound = true; | |
} | |
fromIndex = index - 1; | |
newLineSpot = txt.indexOf('\n', fromIndex); | |
} | |
if (!trailingSpaceFound) { | |
builder.append(txt.substring(fromIndex)); | |
} | |
return builder.toString(); | |
} | |
// private static void test(String i, String o) { | |
// String out = decodeAscii(i); | |
// if (!out.equals(o)) { | |
// throw new AssertionError(String.format("%s to %s instead of %s", i, out, o)); | |
// } | |
// System.out.println(out); | |
// } | |
// private static void testStrip(String input, String expected) { | |
// String out = stripLineBreaks(input); | |
// if (!out.equals(expected)) { | |
// throw new AssertionError(String.format("TS: <%s> to <%s> instead of <%s>", input, out, expected)); | |
// } | |
// System.out.println(out); | |
// } | |
private static String toUtf8(byte[] data) { | |
return new String(data, utf_8); | |
} | |
private static byte[] toByteData(String s) { | |
return s.getBytes(utf_8); | |
} | |
private static final Set<Character> charsToSkip = makeNonSkipChars(); | |
private static Set<Character> makeNonSkipChars() { | |
Set<Character> set = new HashSet<>(); | |
addRange(set, 'a', 'z'); | |
addRange(set, 'A', 'Z'); | |
addRange(set, '0', '9'); | |
set.add('('); | |
set.add(')'); | |
set.add('+'); | |
set.add('-'); | |
set.add('.'); | |
set.add('='); | |
set.add('_'); | |
return set; | |
} | |
private static void addRange(Set<? super Character> cSet, char first, char last) { | |
for (char c = first; c <= last; ++c) { | |
cSet.add(c); | |
} | |
} | |
/** | |
* Convert a JSON string to pretty print version | |
* Source: https://coderwall.com/p/ab5qha/convert-json-string-to-pretty-print-java-gson | |
* | |
* @param jsonString The raw JSON String | |
* @return Prettified String. If an exception is thrown while converting (because it's | |
* not JSON), returns the original String. | |
*/ | |
public static String toPrettyFormat(String jsonString) { | |
JsonParser parser = new JsonParser(); | |
try { | |
JsonObject json = parser.parse(jsonString).getAsJsonObject(); // throws RuntimeExceptions | |
Gson gson = new GsonBuilder().setPrettyPrinting().create(); | |
return gson.toJson(json); | |
} catch (RuntimeException e) { | |
// it's not always a json String. If it's not, just return it. | |
return jsonString; | |
} | |
} | |
/** | |
* Most values arrive as property expressions as P=S, where only S is the JSON String. So we split the value at | |
* the = character, and just prettify what follows it. But we put the whole thing in the pane. If there is no | |
* equal sign, we try to prettify the while String. | |
* @param jsonString The property expression to prettify. | |
* @return A property expression with a prettified value. If the expression is not a property expression, the | |
* prettified String. If it's not a JSON String, we just return it. | |
*/ | |
public static String propertyToPrettyFormat(String jsonString) { | |
// for p=s where p is a property name and s is a JSON String | |
int eSpot = jsonString.indexOf('='); | |
if (eSpot >= 0) { | |
int jSpot = eSpot+1; | |
String head = jsonString.substring(0, jSpot); | |
String tail = jsonString.substring(jSpot); | |
return head+toPrettyFormat(tail); | |
} | |
return toPrettyFormat(jsonString); | |
} | |
public static String chromeToPostman(String source) { | |
StringBuilder builder = new StringBuilder(); | |
StringTokenizer tokenizer = new StringTokenizer(source, "&"); | |
while (tokenizer.hasMoreTokens()) { | |
String token = tokenizer.nextToken(); | |
processChromeToken(token, builder); | |
} | |
return builder.toString(); | |
} | |
private static void processChromeToken(String token, StringBuilder result) { | |
/* | |
Postman x-www-form-urlencoded Bulk Edit: | |
key:value\n | |
Chrome Inspect "View Source" | |
key=value& -- all url encoded utf-8 | |
*/ | |
int eSpot = token.indexOf('='); | |
if (eSpot >= 0) { | |
int vStart = eSpot+1; | |
String head = token.substring(0, eSpot); | |
String tail = token.substring(vStart); | |
result | |
.append(decodeAscii(head)) | |
.append(':') | |
.append(decodeAscii(tail)) | |
.append('\n'); | |
} else { | |
result.append(decodeAscii(token)); | |
} | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Download library:
https://mvnrepository.com/artifact/com.google.code.gson/gson/2.8.5