All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.apple.foundationdb.tuple.StringUtil Maven / Gradle / Ivy

/*
 * StringUtil.java
 *
 * This source file is part of the FoundationDB open source project
 *
 * Copyright 2013-2018 Apple Inc. and the FoundationDB project authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.apple.foundationdb.tuple;

final class StringUtil {
	private static final char SURROGATE_COUNT = Character.MAX_LOW_SURROGATE - Character.MIN_HIGH_SURROGATE + 1;
	private static final char ABOVE_SURROGATES = Character.MAX_VALUE - Character.MAX_LOW_SURROGATE;
	private static final String HIGH_WITHOUT_LOW_ERR_MSG = "malformed UTF-16 string contains high surrogate that is not followed by low surrogate";
	private static final String LOW_WITHOUT_HIGH_ERR_MSG = "malformed UTF-16 string contains low surrogate without prior high surrogate";

	static char adjustForSurrogates(char c, String s, int pos) {
		if(c > Character.MAX_LOW_SURROGATE) {
			return (char)(c - SURROGATE_COUNT);
		}
		else {
			// Validate the UTF-16 string as this can do weird things on invalid strings
			if(Character.isHighSurrogate(c) && (pos + 1 >= s.length() || !Character.isLowSurrogate(s.charAt(pos + 1)))) {
				throw new IllegalArgumentException(HIGH_WITHOUT_LOW_ERR_MSG);
			}
			else if(Character.isLowSurrogate(c) && (pos == 0 || !Character.isHighSurrogate(s.charAt(pos - 1)))) {
				throw new IllegalArgumentException(LOW_WITHOUT_HIGH_ERR_MSG);
			}
			return (char)(c + ABOVE_SURROGATES);

		}
	}

	// Validates that the string is well-formed UTF-16 by making sure every high surrogate is followed by a low-surrogate
	static void validate(String s) {
		final int strLength = s.length();
		int i = 0;
		while(i < strLength) {
			char c = s.charAt(i);
			if(Character.isHighSurrogate(c)) {
				if(i + 1 >= strLength || !Character.isLowSurrogate(s.charAt(i + 1))) {
					throw new IllegalArgumentException(HIGH_WITHOUT_LOW_ERR_MSG);
				}
				i += 2;
			}
			else if(Character.isLowSurrogate(c)) {
				throw new IllegalArgumentException(LOW_WITHOUT_HIGH_ERR_MSG);
			}
			else {
				i++;
			}
		}
	}

	// Compare two strings based on their UTF-8 code point values. Note that Java stores strings
	// using UTF-16. However, {@link Tuple}s are encoded using UTF-8. Using unsigned byte comparison,
	// UTF-8 strings will sort based on their Unicode codepoints. However, UTF-16 strings almost,
	// but not quite, sort that way. This can be addressed by fixing up surrogates. There are 0x800 surrogate
	// values and about 0x2000 code points above the maximum surrogate value. For anything that is a surrogate,
	// shift it up by 0x2000, and anything that is above the maximum surrogate value, shift it down by 0x800.
	// This makes all surrogates sort after all non-surrogates.
	//
	// See: https://ssl.icu-project.org/docs/papers/utf16_code_point_order.html
	static int compareUtf8(String s1, String s2) {
		// Ignore common prefix at the beginning which will compare equal regardless of encoding
		final int s1Length = s1.length();
		final int s2Length = s2.length();
		int pos = 0;
		while(pos < s1Length && pos < s2Length && s1.charAt(pos) == s2.charAt(pos)) {
			pos++;
		}
		if(pos >= s1Length || pos >= s2Length) {
			// One string is the prefix of another, so return based on length.
			return Integer.compare(s1Length, s2Length);
		}
		// Compare first different character
		char c1 = s1.charAt(pos);
		char c2 = s2.charAt(pos);
		// Apply "fix up" for surrogates
		if(c1 >= Character.MIN_HIGH_SURROGATE) {
			c1 = adjustForSurrogates(c1, s1, pos);
		}
		if(c2 >= Character.MIN_HIGH_SURROGATE) {
			c2 = adjustForSurrogates(c2, s2, pos);
		}
		return Character.compare(c1, c2);
	}

	static int packedSize(String s) {
		final int strLength = s.length();
		int size = 0;
		int pos = 0;

		while(pos < strLength) {
			char c = s.charAt(pos);
			if(c == '\0') {
				// Null is encoded as \x00\xff
				size += 2;
			}
			else if(c <= 0x7f) {
				// ASCII code point. Only 1 byte.
				size += 1;
			}
			else if(c <= 0x07ff) {
				// 2 byte code point
				size += 2;
			}
			else if(Character.isHighSurrogate(c)) {
				if(pos + 1 < s.length() && Character.isLowSurrogate(s.charAt(pos + 1))) {
					// High surrogate followed by low surrogate means the code point
					// is between U+10000 and U+10FFFF, so it requires 4 bytes.
					size += 4;
					pos += 1;
				}
				else {
					throw new IllegalArgumentException(HIGH_WITHOUT_LOW_ERR_MSG);
				}
			}
			else if(Character.isLowSurrogate(c)) {
				throw new IllegalArgumentException(LOW_WITHOUT_HIGH_ERR_MSG);
			}
			else {
				// 3 byte code point
				size += 3;
			}
			pos += 1;
		}

		return size;
	}

	private StringUtil() {}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy