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

org.hl7.fhir.utilities.Utilities Maven / Gradle / Ivy

There is a newer version: 6.4.1
Show newest version
package org.hl7.fhir.utilities;

import static org.apache.commons.lang3.StringUtils.isBlank;

import java.io.BufferedInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.FilenameFilter;
import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.net.URI;
import java.net.URLDecoder;
import java.net.URLEncoder;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardCopyOption;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;
import java.util.UUID;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;

import javax.annotation.Nullable;

/*
  Copyright (c) 2011+, HL7, Inc.
  All rights reserved.

  Redistribution and use in source and binary forms, with or without modification, 
  are permitted provided that the following conditions are met:

 * Redistributions of source code must retain the above copyright notice, this 
     list of conditions and the following disclaimer.
 * Redistributions in binary form must reproduce the above copyright notice, 
     this list of conditions and the following disclaimer in the documentation 
     and/or other materials provided with the distribution.
 * Neither the name of HL7 nor the names of its contributors may be used to 
     endorse or promote products derived from this software without specific 
     prior written permission.

  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND 
  ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 
  WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 
  IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, 
  INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 
  NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 
  PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, 
  WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 
  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 
  POSSIBILITY OF SUCH DAMAGE.

 */


import org.apache.commons.io.FileUtils;
import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.utilities.FileNotifier.FileNotifier2;
import org.hl7.fhir.utilities.Utilities.CaseInsensitiveSorter;
import org.hl7.fhir.utilities.filesystem.CSFile;
import org.hl7.fhir.utilities.filesystem.ManagedFileAccess;
import org.hl7.fhir.utilities.settings.FhirSettings;

public class Utilities {

  private static final String UUID_REGEX = "[0-9a-f]{8}\\-[0-9a-f]{4}\\-[0-9a-f]{4}\\-[0-9a-f]{4}\\-[0-9a-f]{12}";
  private static final String OID_REGEX = "[0-2](\\.(0|[1-9][0-9]*))+";
  static final String C_TEMP_DIR = "c:\\temp";

  /**
   * Returns the plural form of the word in the string.
   * 

* Examples: * *

   *   inflector.pluralize("post")               #=> "posts"
   *   inflector.pluralize("octopus")            #=> "octopi"
   *   inflector.pluralize("sheep")              #=> "sheep"
   *   inflector.pluralize("words")              #=> "words"
   *   inflector.pluralize("the blue mailman")   #=> "the blue mailmen"
   *   inflector.pluralize("CamelOctopus")       #=> "CamelOctopi"
   * 
*

*

*

* Note that if the {@link Object#toString()} is called on the supplied object, so this method works for non-strings, too. * * @param word the word that is to be pluralized. * @return the pluralized form of the word, or the word itself if it could not be pluralized * @see Inflector#singularize(Object) */ public static String pluralizeMe(String word) { Inflector inf = new Inflector(); return inf.pluralize(word); } public static String pluralize(String word, int count) { if (count == 1) return word; Inflector inf = new Inflector(); return inf.pluralize(word); } public static String singularise(String word) { Inflector inf = new Inflector(); return inf.singularize(word); } public static boolean isInteger(String string) { if (isBlank(string)) { return false; } String value = string.startsWith("-") ? string.substring(1) : string; if (Utilities.noString(value)) { return false; } for (char next : value.toCharArray()) { if (!Character.isDigit(next)) { return false; } } // check bounds -2,147,483,648..2,147,483,647 if (value.length() > 10) return false; if (string.startsWith("-")) { if (value.length() == 10 && string.compareTo("2147483648") > 0) return false; } else { if (value.length() == 10 && string.compareTo("2147483647") > 0) return false; } return true; } public static boolean isLong(String string) { if (isBlank(string)) { return false; } String value = string.startsWith("-") ? string.substring(1) : string; for (char next : value.toCharArray()) { if (!Character.isDigit(next)) { return false; } } // check bounds -9,223,372,036,854,775,808 to +9,223,372,036,854,775,807 if (value.length() > 20) return false; if (string.startsWith("-")) { if (value.length() == 20 && string.compareTo("9223372036854775808") > 0) return false; } else { if (value.length() == 20 && string.compareTo("9223372036854775807") > 0) return false; } return true; } public static boolean isHex(String string) { try { int i = Integer.parseInt(string, 16); return i != i + 1; } catch (Exception e) { return false; } } public enum DecimalStatus { BLANK, SYNTAX, RANGE, OK } public static boolean isDecimal(String value, boolean allowExponent, boolean allowLeadingZero) { DecimalStatus ds = checkDecimal(value, allowExponent, true); return ds == DecimalStatus.OK || ds == DecimalStatus.RANGE; } public static boolean isDecimal(String value, boolean allowExponent) { DecimalStatus ds = checkDecimal(value, allowExponent, false); return ds == DecimalStatus.OK || ds == DecimalStatus.RANGE; } public static DecimalStatus checkDecimal(String value, boolean allowExponent, boolean allowLeadingZero) { if (isBlank(value)) { return DecimalStatus.BLANK; } // check for leading zeros if (!allowLeadingZero) { if (value.startsWith("0") && !"0".equals(value) && !value.startsWith("0.")) return DecimalStatus.SYNTAX; if (value.startsWith("-0") && !"-0".equals(value) && !value.startsWith("-0.")) return DecimalStatus.SYNTAX; if (value.startsWith("+0") && !"+0".equals(value) && !value.startsWith("+0.")) return DecimalStatus.SYNTAX; } // check for trailing dot if (value.endsWith(".")) { return DecimalStatus.SYNTAX; } boolean havePeriod = false; boolean haveExponent = false; boolean haveSign = false; boolean haveDigits = false; int preDecLength = 0; int postDecLength = 0; int exponentLength = 0; int length = 0; for (char next : value.toCharArray()) { if (next == '.') { if (!haveDigits || havePeriod || haveExponent) return DecimalStatus.SYNTAX; havePeriod = true; preDecLength = length; length = 0; } else if (next == '-' || next == '+') { if (haveDigits || haveSign) return DecimalStatus.SYNTAX; haveSign = true; } else if (next == 'e' || next == 'E') { if (!haveDigits || haveExponent || !allowExponent) return DecimalStatus.SYNTAX; haveExponent = true; haveSign = false; haveDigits = false; if (havePeriod) postDecLength = length; else preDecLength = length; length = 0; } else if (!Character.isDigit(next)) { return DecimalStatus.SYNTAX; } else { haveDigits = true; length++; } } if (haveExponent && !haveDigits) return DecimalStatus.SYNTAX; if (haveExponent) exponentLength = length; else if (havePeriod) postDecLength = length; else preDecLength = length; // now, bounds checking - these are arbitrary if (exponentLength > 4) return DecimalStatus.RANGE; if (preDecLength + postDecLength > 18) return DecimalStatus.RANGE; return DecimalStatus.OK; } public static String camelCase(String value) { return new Inflector().camelCase(value.trim().replace(" ", "_"), false); } public static String upperCamelCase(String value) { return new Inflector().upperCamelCase(value.trim().replace(" ", "_")); } public static String escapeXml(String doco) { if (doco == null) return ""; StringBuilder b = new StringBuilder(); for (char c : doco.toCharArray()) { if (c == '<') b.append("<"); else if (c == '>') b.append(">"); else if (c == '&') b.append("&"); else if (c == '"') b.append("""); else b.append(c); } return b.toString(); } public static String titleize(String s) { StringBuilder b = new StringBuilder(); boolean up = true; for (char c : s.toCharArray()) { if (up) b.append(Character.toUpperCase(c)); else b.append(c); up = c == ' '; } return b.toString(); } public static String capitalize(String s) { if (s == null) return null; if (s.length() == 0) return s; if (s.length() == 1) return s.toUpperCase(); return s.substring(0, 1).toUpperCase() + s.substring(1); } public static void copyDirectory(String sourceFolder, String destFolder, FileNotifier notifier) throws IOException, FHIRException { CSFile src = ManagedFileAccess.csfile(sourceFolder); if (!src.exists()) throw new FHIRException("Folder " + sourceFolder + " not found"); File dst = ManagedFileAccess.file(destFolder); if(!dst.getCanonicalFile().getName().equals(dst.getName())) { File tmp = ManagedFileAccess.file(destFolder+System.currentTimeMillis()); if (!dst.renameTo(tmp)) { throw new IOException("fixing case from "+dst.getCanonicalFile().getName()+" to "+tmp.getName()+" failed"); } if (!tmp.renameTo(dst)) { throw new IOException("fixing case from "+tmp.getCanonicalFile().getName()+" to "+dst.getName()+" failed"); } } else if (!dst.exists()) { createDirectory(destFolder); } String[] files = src.list(); for (String f : files) { if (ManagedFileAccess.csfile(sourceFolder + File.separator + f).isDirectory()) { if (!f.startsWith(".")) { // ignore .git files... copyDirectory(sourceFolder + File.separator + f, destFolder + File.separator + f, notifier); } } else { if (notifier != null) notifier.copyFile(sourceFolder + File.separator + f, destFolder + File.separator + f); copyFile(ManagedFileAccess.csfile(sourceFolder + File.separator + f), new /*CS*/File(destFolder + File.separator + f)); // case doesn't have to match on the target } } } public static void copyDirectory2(String sourceFolder, String destFolder, FileNotifier2 notifier) throws IOException, FHIRException { CSFile src = ManagedFileAccess.csfile(sourceFolder); if (!src.exists()) throw new FHIRException("Folder " + sourceFolder + " not found"); createDirectory(destFolder); String[] files = src.list(); for (String f : files) { if (ManagedFileAccess.csfile(sourceFolder + File.separator + f).isDirectory()) { if (!f.startsWith(".")) { // ignore .git files... boolean doCopy = notifier != null ? notifier.copyFolder(sourceFolder + File.separator + f, destFolder + File.separator + f) : true; if (doCopy) { copyDirectory2(sourceFolder + File.separator + f, destFolder + File.separator + f, notifier); } } } else { boolean doCopy = notifier != null ? notifier.copyFile(sourceFolder + File.separator + f, destFolder + File.separator + f) : true; if (doCopy) { copyFile(ManagedFileAccess.csfile(sourceFolder + File.separator + f), ManagedFileAccess.csfile(destFolder + File.separator + f)); } } } } public static void copyFile(String source, String dest) throws IOException { copyFile(ManagedFileAccess.file(source), ManagedFileAccess.file(dest)); } public static void copyFile(File sourceFile, File destFile) throws IOException { if (!destFile.exists()) { if (!ManagedFileAccess.csfile(destFile.getParent()).exists()) { createDirectory(destFile.getParent()); } destFile.createNewFile(); } else if (!destFile.getCanonicalFile().getName().equals(destFile.getName())) { // case mismatch destFile.delete(); destFile.createNewFile(); } FileInputStream source = null; FileOutputStream destination = null; try { source = ManagedFileAccess.inStream(sourceFile); destination = ManagedFileAccess.outStream(destFile); destination.getChannel().transferFrom(source.getChannel(), 0, source.getChannel().size()); } finally { if (source != null) { source.close(); } if (destination != null) { destination.close(); } } } public static boolean checkFolder(String dir, List errors) throws IOException { if (!ManagedFileAccess.csfile(dir).exists()) { errors.add("Unable to find directory " + dir); return false; } else { return true; } } public static boolean checkFile(String purpose, String dir, String file, List errors) throws IOException { if (!ManagedFileAccess.csfile(dir + file).exists()) { if (errors != null) errors.add("Unable to find " + purpose + " file " + file + " in " + dir); return false; } else { return true; } } public static String asCSV(List strings) { StringBuilder s = new StringBuilder(); boolean first = true; for (String n : strings) { if (!first) s.append(","); s.append(n); first = false; } return s.toString(); } public static String asHtmlBr(String prefix, List strings) { StringBuilder s = new StringBuilder(); boolean first = true; for (String n : strings) { if (!first) s.append("
"); s.append(prefix); s.append(n); first = false; } return s.toString(); } public static void clearDirectory(String folder, String... exemptions) throws IOException { File dir = ManagedFileAccess.file(folder); if (dir.exists()) { if (exemptions.length == 0) FileUtils.cleanDirectory(dir); else { String[] files = ManagedFileAccess.csfile(folder).list(); if (files != null) { for (String f : files) { if (!existsInList(f, exemptions)) { File fh = ManagedFileAccess.csfile(folder + File.separatorChar + f); if (fh.isDirectory()) clearDirectory(fh.getAbsolutePath()); fh.delete(); } } } } } } public static File createDirectory(String path) throws IOException { ManagedFileAccess.csfile(path).mkdirs(); return ManagedFileAccess.file(path); } public static String changeFileExt(String name, String ext) { if (name.lastIndexOf('.') > -1) return name.substring(0, name.lastIndexOf('.')) + ext; else return name + ext; } public static String cleanupTextString(String contents) { if (contents == null || contents.trim().equals("")) return null; else return contents.trim(); } public static boolean noString(String v) { return v == null || v.equals(""); } public static void bytesToFile(byte[] content, String filename) throws IOException { FileOutputStream out = ManagedFileAccess.outStream(filename); out.write(content); out.close(); } public static String appendSlash(String definitions) { return definitions.endsWith(File.separator) ? definitions : definitions + File.separator; } public static String appendForwardSlash(String definitions) { if (definitions == null) { return "/"; } return definitions.endsWith("/") ? definitions : definitions + "/"; } public static String fileTitle(String file) throws IOException { if (file == null) return null; String s = ManagedFileAccess.file(file).getName(); return s.indexOf(".") == -1 ? s : s.substring(0, s.indexOf(".")); } public static String systemEol() { return System.getProperty("line.separator"); } public static String normaliseEolns(String value) { return value.replace("\r\n", "\r").replace("\n", "\r").replace("\r", "\r\n"); } public static String unescapeXml(String xml) throws FHIRException { if (xml == null) return null; StringBuilder b = new StringBuilder(); int i = 0; while (i < xml.length()) { if (xml.charAt(i) == '&') { StringBuilder e = new StringBuilder(); i++; while (xml.charAt(i) != ';') { e.append(xml.charAt(i)); i++; } if (e.toString().equals("lt")) b.append("<"); else if (e.toString().equals("gt")) b.append(">"); else if (e.toString().equals("amp")) b.append("&"); else if (e.toString().equals("quot")) b.append("\""); else if (e.toString().equals("mu")) b.append((char) 956); else throw new FHIRException("unknown XML entity \"" + e.toString() + "\""); } else b.append(xml.charAt(i)); i++; } return b.toString(); } public static String unescapeJson(String json) throws FHIRException { if (json == null) return null; StringBuilder b = new StringBuilder(); int i = 0; while (i < json.length()) { if (json.charAt(i) == '\\') { i++; char ch = json.charAt(i); switch (ch) { case '"': b.append('b'); break; case '\\': b.append('\\'); break; case '/': b.append('/'); break; case 'b': b.append('\b'); break; case 'f': b.append('\f'); break; case 'n': b.append('\n'); break; case 'r': b.append('\r'); break; case 't': b.append('\t'); break; case 'u': String hex = json.substring(i + 1, i + 5); b.append(Character.toString(Integer.parseInt(hex, 16))); break; default: throw new FHIRException("Unknown JSON escape \\" + ch); } } else b.append(json.charAt(i)); i++; } return b.toString(); } public static boolean isPlural(String word) { word = word.toLowerCase(); if ("restricts".equals(word) || "contains".equals(word) || "data".equals(word) || "specimen".equals(word) || "replaces".equals(word) || "addresses".equals(word) || "supplementalData".equals(word) || "instantiates".equals(word) || "imports".equals(word) || "covers".equals(word)) return false; Inflector inf = new Inflector(); return !inf.singularize(word).equals(word); } public static String padRight(String src, char c, int len) { StringBuilder s = new StringBuilder(); if (src != null) { s.append(src); for (int i = 0; i < len - src.length(); i++) s.append(c); } return s.toString(); } public static String padLeft(String src, char c, int len) { StringBuilder s = new StringBuilder(); for (int i = 0; i < len - src.length(); i++) s.append(c); s.append(src); return s.toString(); } /** * Composes a path string using by concatenating the passed arguments. * * This method enables all checks for unintended path locations. * * @param args * @return * @throws IOException * @see PathBuilder#buildPath(String...) */ public static String path(String... args) throws IOException { return PathBuilder.getPathBuilder().buildPath(args); } public static File pathFile(String... args) throws IOException { return ManagedFileAccess.file(PathBuilder.getPathBuilder().buildPath(args)); } public static String path(File f, String... args) throws IOException { String[] a = new String[args.length+1]; a[0] = f.getAbsolutePath(); for (int i = 0; i < args.length; i++) { a[i+1] = args[i]; } return PathBuilder.getPathBuilder().buildPath(a); } /** * Composes a path string using by concatenating the passed arguments. * * This method does not check for unintentional access to areas of the file * system outside of the first entry. ONLY USE THIS METHOD IN CASES WHERE YOU * ARE CERTAIN THE COMPOSED PATH IS NOT MALICIOUS. * * @param args * @return * @throws IOException * * @see PathBuilder#buildPath(String...) */ @Deprecated public static String uncheckedPath(String... args) throws IOException { return PathBuilder.getPathBuilder() .withRequireNonRootFirstEntry(false) .withRequireNonNullNonEmptyFirstEntry(false) .withRequirePathIsChildOfTarget(false) .buildPath(args); } public static String pathURL(String... args) { StringBuilder s = new StringBuilder(); boolean d = false; for (String arg : args) { if (arg != null) { if (!d) d = !noString(arg); else if (s.toString() != null && !s.toString().endsWith("/") && !arg.startsWith("/")) s.append("/"); s.append(arg); } } return s.toString(); } public static String nmtokenize(String cs) { if (cs == null) return ""; StringBuilder s = new StringBuilder(); for (int i = 0; i < cs.length(); i++) { char c = cs.charAt(i); if ((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9') || c == '-' || c == '_') s.append(c); else if (c != ' ') s.append("." + Integer.toString(c)); } return s.toString(); } public static String javaTokenize(String cs, boolean capFirst) { if (cs == null) return ""; StringBuilder s = new StringBuilder(); boolean upcase = capFirst; for (int i = 0; i < cs.length(); i++) { char c = cs.charAt(i); if ((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9') || c == '_') { if (upcase) { s.append(Character.toUpperCase(c)); upcase = false; } else { s.append(c); } } else { upcase = true; } } String res = s.toString(); if (isJavaReservedWord(res)) { return "_"+res; } else { return res; } } public static boolean isJavaReservedWord(String word) { if (word.equals("abstract")) return true; if (word.equals("assert")) return true; if (word.equals("boolean")) return true; if (word.equals("break")) return true; if (word.equals("byte")) return true; if (word.equals("case")) return true; if (word.equals("catch")) return true; if (word.equals("char")) return true; if (word.equals("class")) return true; if (word.equals("const")) return true; if (word.equals("continue")) return true; if (word.equals("default")) return true; if (word.equals("double")) return true; if (word.equals("do")) return true; if (word.equals("else")) return true; if (word.equals("enum")) return true; if (word.equals("extends")) return true; if (word.equals("false")) return true; if (word.equals("final")) return true; if (word.equals("finally")) return true; if (word.equals("float")) return true; if (word.equals("for")) return true; if (word.equals("goto")) return true; if (word.equals("if")) return true; if (word.equals("implements")) return true; if (word.equals("import")) return true; if (word.equals("instanceof")) return true; if (word.equals("int")) return true; if (word.equals("interface")) return true; if (word.equals("long")) return true; if (word.equals("native")) return true; if (word.equals("new")) return true; if (word.equals("null")) return true; if (word.equals("package")) return true; if (word.equals("private")) return true; if (word.equals("protected")) return true; if (word.equals("public")) return true; if (word.equals("return")) return true; if (word.equals("short")) return true; if (word.equals("static")) return true; if (word.equals("strictfp")) return true; if (word.equals("super")) return true; if (word.equals("switch")) return true; if (word.equals("synchronized")) return true; if (word.equals("this")) return true; if (word.equals("throw")) return true; if (word.equals("throws")) return true; if (word.equals("transient")) return true; if (word.equals("true")) return true; if (word.equals("try")) return true; if (word.equals("void")) return true; if (word.equals("volatile")) return true; if (word.equals("while")) return true; if (word.equals("Exception")) return true; return false; } public static boolean isToken(String tail) { if (tail == null || tail.length() == 0) return false; boolean result = isAlphabetic(tail.charAt(0)); for (int i = 1; i < tail.length(); i++) { result = result && (isAlphabetic(tail.charAt(i)) || isDigit(tail.charAt(i)) || (tail.charAt(i) == '_') || (tail.charAt(i) == '[') || (tail.charAt(i) == ']')); } return result; } public static boolean isTokenChar(char ch) { return isAlphabetic(ch) || (ch == '_'); } public static boolean isDigit(char c) { return (c >= '0') && (c <= '9'); } public static boolean isAlphabetic(char c) { return ((c >= 'a') && (c <= 'z')) || ((c >= 'A') && (c <= 'Z')); } public static String getDirectoryForFile(String filepath) throws IOException { File f = ManagedFileAccess.file(filepath); return f.getParent(); } public static String getDirectoryForURL(String url) { return url.contains("/") && url.lastIndexOf("/") > 10 ? url.substring(0, url.lastIndexOf("/")) : url; } public static String appendPeriod(String s) { if (Utilities.noString(s)) return s; s = s.trim(); if (s.endsWith(".") || s.endsWith("?")) return s; return s + "."; } public static String removePeriod(String s) { if (Utilities.noString(s)) return s; if (s.endsWith(".")) return s.substring(0, s.length() - 1); return s; } public static String stripBOM(String string) { return string == null ? null : string.replace("\uFEFF", ""); } public static String oidTail(String id) { if (id == null || !id.contains(".")) return id; return id.substring(id.lastIndexOf(".") + 1); } public static String oidRoot(String id) { if (id == null || !id.contains(".")) return id; return id.substring(0, id.indexOf(".")); } public static String escapeJava(String doco) { if (doco == null) return ""; StringBuilder b = new StringBuilder(); for (char c : doco.toCharArray()) { if (c == '\r') b.append("\\r"); else if (c == '\n') b.append("\\n"); else if (c == '"') b.append("\\\""); else if (c == '\\') b.append("\\\\"); else b.append(c); } return b.toString(); } public static String[] splitByCamelCase(String name) { List parts = new ArrayList(); StringBuilder b = new StringBuilder(); for (int i = 0; i < name.length(); i++) { if (i > 0 && Character.isUpperCase(name.charAt(i))) { parts.add(b.toString()); b = new StringBuilder(); } b.append(Character.toLowerCase(name.charAt(i))); } parts.add(b.toString()); return parts.toArray(new String[]{}); } public static String encodeUri(String v) { return v.replace("%", "%25").replace(" ", "%20").replace("?", "%3F").replace("=", "%3D").replace("|", "%7C").replace("+", "%2B"); } public static String normalize(String s) { if (noString(s)) return null; StringBuilder b = new StringBuilder(); boolean isWhitespace = false; for (int i = 0; i < s.length(); i++) { char c = s.charAt(i); if (!isWhitespace(c)) { b.append(Character.toLowerCase(c)); isWhitespace = false; } else if (!isWhitespace) { if (c == '\r' || c == '\n') { b.append('\n'); } else { b.append(' '); } isWhitespace = true; } } return b.toString().trim(); } public static String normalizeSameCase(String s) { if (noString(s)) return null; StringBuilder b = new StringBuilder(); boolean isWhitespace = false; for (int i = 0; i < s.length(); i++) { char c = s.charAt(i); if (!Character.isWhitespace(c)) { b.append(c); isWhitespace = false; } else if (!isWhitespace) { b.append(' '); isWhitespace = true; } } return b.toString().trim(); } public static void copyFileToDirectory(File source, File destDir) throws IOException { copyFile(source, ManagedFileAccess.file(path(destDir.getAbsolutePath(), source.getName()))); } public static String URLEncode(String string) { try { return URLEncoder.encode(string, "UTF-8"); } catch (UnsupportedEncodingException e) { throw new Error(e.getMessage()); } } public static String URLDecode(String ref) { try { return URLDecoder.decode(ref, "UTF-8"); } catch (UnsupportedEncodingException e) { throw new Error(e.getMessage()); } } public static boolean charInSet(char value, char... array) { for (int i : array) if (value == i) return true; return false; } public static boolean charInRange(char ch, char a, char z) { return ch >= a && ch <= z; } public static boolean existsInList(String value, List array) { if (value == null) return false; for (String s : array) if (value.equals(s)) return true; return false; } public static boolean containsInList(String value, String... array) { if (value == null) return false; for (String s : array) if (value.contains(s)) return true; return false; } public static boolean existsInList(String value, String... array) { if (value == null) return false; for (String s : array) if (value.equals(s)) return true; return false; } public static boolean existsInListTrimmed(String value, String... array) { if (value == null) return false; for (String s : array) if (value.equals(s.trim())) return true; return false; } public static boolean existsInList(int value, int... array) { for (int i : array) if (value == i) return true; return false; } public static boolean existsInListNC(String value, String... array) { for (String s : array) if (value.equalsIgnoreCase(s)) return true; return false; } public static String stringJoin(String sep, String... array) { CommaSeparatedStringBuilder b = new CommaSeparatedStringBuilder(sep); for (String s : array) if (!noString(s)) b.append(s); return b.toString(); } public static String getFileNameForName(String name) { return name.toLowerCase(); } public static void deleteTempFiles() throws IOException { File file = createTempFile("test", "test"); String folder = getDirectoryForFile(file.getAbsolutePath()); String[] list = ManagedFileAccess.file(folder).list(new FilenameFilter() { public boolean accept(File dir, String name) { return name.startsWith("ohfu-"); } }); if (list != null) { for (String n : list) { ManagedFileAccess.file(path(folder, n)).delete(); } } } public static File createTempFile(String prefix, String suffix) throws IOException { // this allows use to eaily identify all our dtemp files and delete them, since delete on Exit doesn't really work. File file = File.createTempFile("ohfu-" + prefix, suffix); file.deleteOnExit(); return file; } public static boolean isAsciiChar(char ch) { return ch >= ' ' && ch <= '~'; } public static String makeUuidLC() { return UUID.randomUUID().toString().toLowerCase(); } public static String makeUuidUrn() { return "urn:uuid:" + UUID.randomUUID().toString().toLowerCase(); } public static boolean isURL(String s) { boolean ok = s.matches("^http(s{0,1})://[a-zA-Z0-9_/\\-\\.]+\\.([A-Za-z/]{2,5})[a-zA-Z0-9_/\\&\\?\\=\\-\\.\\~\\%]*"); return ok; } public static String escapeCSV(String value) { if (value == null) return ""; StringBuilder b = new StringBuilder(); for (char c : value.toCharArray()) { if (c == '"') b.append("\"\""); else if (isWhitespace(c)) b.append(" "); else b.append(c); } return b.toString(); } public static String escapeJson(String value) { return escapeJson(value, true); } public static String escapeJson(String value, boolean escapeUnicodeWhitespace) { if (value == null) return ""; StringBuilder b = new StringBuilder(); for (char c : value.toCharArray()) { if (c == '\r') b.append("\\r"); else if (c == '\n') b.append("\\n"); else if (c == '\t') b.append("\\t"); else if (c == '"') b.append("\\\""); else if (c == '\\') b.append("\\\\"); else if (c == ' ') b.append(" "); else if ((c == '\r' || c == '\n') || (isWhitespace(c) && escapeUnicodeWhitespace)) { b.append("\\u"+Utilities.padLeft(Integer.toHexString(c), '0', 4)); } else if (((int) c) < 32) b.append("\\u" + Utilities.padLeft(Integer.toHexString(c), '0', 4)); else b.append(c); } return b.toString(); } public static String humanize(String code) { StringBuilder b = new StringBuilder(); boolean lastBreak = true; for (char c : code.toCharArray()) { if (Character.isLetter(c)) { if (lastBreak) b.append(Character.toUpperCase(c)); else { if (Character.isUpperCase(c)) b.append(" "); b.append(c); } lastBreak = false; } else { b.append(" "); lastBreak = true; } } if (b.length() == 0) return code; else return b.toString(); } public static String uncapitalize(String s) { if (s == null) return null; if (s.length() == 0) return s; if (s.length() == 1) return s.toLowerCase(); return s.substring(0, 1).toLowerCase() + s.substring(1); } public static int charCount(String s, char c) { int res = 0; for (char ch : s.toCharArray()) if (ch == c) res++; return res; } public static boolean isOid(String cc) { return cc.matches(OID_REGEX); } public static boolean equals(String one, String two) { if (one == null && two == null) return true; if (one == null || two == null) return false; return one.equals(two); } public static void deleteAllFiles(String folder, String type) throws IOException { File src = ManagedFileAccess.file(folder); String[] files = src.list(); for (String f : files) { if (ManagedFileAccess.file(folder + File.separator + f).isDirectory()) { deleteAllFiles(folder + File.separator + f, type); } else if (f.endsWith(type)) { ManagedFileAccess.file(folder + File.separator + f).delete(); } } } public static boolean compareIgnoreWhitespace(File f1, File f2) throws IOException { InputStream in1 = null; InputStream in2 = null; try { in1 = new BufferedInputStream(ManagedFileAccess.inStream(f1)); in2 = new BufferedInputStream(ManagedFileAccess.inStream(f2)); int expectedByte = in1.read(); while (expectedByte != -1) { boolean w1 = Character.isWhitespace(expectedByte); if (w1) while (Character.isWhitespace(expectedByte)) expectedByte = in1.read(); int foundByte = in2.read(); if (w1) { if (!Character.isWhitespace(foundByte)) return false; while (Character.isWhitespace(foundByte)) foundByte = in2.read(); } if (expectedByte != foundByte) return false; expectedByte = in1.read(); } if (in2.read() != -1) { return false; } return true; } finally { if (in1 != null) { try { in1.close(); } catch (IOException e) { } } if (in2 != null) { try { in2.close(); } catch (IOException e) { } } } } public static boolean compareIgnoreWhitespace(String fn1, String fn2) throws IOException { return compareIgnoreWhitespace(ManagedFileAccess.file(fn1), ManagedFileAccess.file(fn2)); } public static boolean isAbsoluteUrl(String ref) { if (ref != null && ref.contains(":")) { String scheme = ref.substring(0, ref.indexOf(":")); String details = ref.substring(ref.indexOf(":")+1); return (existsInList(scheme, "http", "https", "urn", "file:") || (isToken(scheme) && scheme.equals(scheme.toLowerCase())) || Utilities.startsWithInList(ref, "urn:iso:", "urn:iso-iec:", "urn:iso-cie:", "urn:iso-astm:", "urn:iso-ieee:", "urn:iec:")) && details != null && details.length() > 0 && !details.contains(" "); // rfc5141 } return false; } public static boolean isAbsoluteUrlLinkable(String ref) { if (ref != null && ref.contains(":")) { String scheme = ref.substring(0, ref.indexOf(":")); String details = ref.substring(ref.indexOf(":")+1); return (existsInList(scheme, "http", "https", "ftp")) && details != null && details.length() > 0 && !details.contains(" "); // rfc5141 } return false; } public static boolean equivalent(String l, String r) { if (Utilities.noString(l) && Utilities.noString(r)) return true; if (Utilities.noString(l) || Utilities.noString(r)) return false; return l.toLowerCase().equals(r.toLowerCase()); } public static boolean equivalentNumber(String l, String r) { if (Utilities.noString(l) && Utilities.noString(r)) return true; if (Utilities.noString(l) || Utilities.noString(r)) return false; if (!Utilities.isDecimal(l, true) || !Utilities.isDecimal(r, true)) return false; BigDecimal dl = new BigDecimal(l); BigDecimal dr = new BigDecimal(r); if (dl.scale() < dr.scale()) { dr = dr.setScale(dl.scale(), RoundingMode.HALF_UP); } else if (dl.scale() > dr.scale()) { dl = dl.setScale(dr.scale(), RoundingMode.HALF_UP); } return dl.equals(dr); } public static String getFileExtension(String fn) { return fn.contains(".") ? fn.substring(fn.lastIndexOf(".") + 1) : ""; } public static String unCamelCaseKeepCapitals(String name) { StringBuilder b = new StringBuilder(); boolean first = true; for (char c : name.toCharArray()) { if (Character.isUpperCase(c)) { if (!first) b.append(" "); b.append(c); } else b.append(c); first = false; } return b.toString(); } public static String unCamelCase(String name) { StringBuilder b = new StringBuilder(); boolean first = true; for (char c : name.toCharArray()) { if (Character.isUpperCase(c)) { if (!first) b.append(" "); b.append(Character.toLowerCase(c)); } else b.append(c); first = false; } return b.toString(); } public static boolean isAbsoluteFileName(String source) { if (isWindows()) return (source.length() > 2 && source.charAt(1) == ':') || source.startsWith("\\\\"); else return source.startsWith("//"); } public static boolean isWindows() { return System.getProperty("os.name").startsWith("Windows"); } public static String splitLineForLength(String line, int prefixLength, int indent, int allowedLength) { List list = new ArrayList(); while (prefixLength + line.length() > allowedLength) { int i = allowedLength - (list.size() == 0 ? prefixLength : indent); while (i > 0 && line.charAt(i) != ' ') i--; if (i == 0) break; list.add(line.substring(0, i)); line = line.substring(i + 1); } list.add(line); StringBuilder b = new StringBuilder(); boolean first = true; for (String s : list) { if (first) first = false; else b.append("\r\n" + padLeft("", ' ', indent)); b.append(s); } return b.toString(); } public static int countFilesInDirectory(String dirName) throws IOException { File dir = ManagedFileAccess.file(dirName); if (dir.exists() == false) { return 0; } int i = 0; for (File f : dir.listFiles()) if (!f.isDirectory()) i++; return i; } public static String makeId(String name) { StringBuilder b = new StringBuilder(); for (char ch : name.toCharArray()) { if (ch >= 'a' && ch <= 'z') b.append(ch); else if (ch >= 'A' && ch <= 'Z') b.append(ch); else if (ch >= '0' && ch <= '9') b.append(ch); else if (ch == '-' || ch == '.') b.append(ch); } return b.toString(); } public interface FileVisitor { void visitFile(File file) throws FileNotFoundException, IOException; } public static void visitFiles(String folder, String extension, FileVisitor visitor) throws FileNotFoundException, IOException { visitFiles(ManagedFileAccess.file(folder), extension, visitor); } public static void visitFiles(File folder, String extension, FileVisitor visitor) throws FileNotFoundException, IOException { for (File file : folder.listFiles()) { if (file.isDirectory()) visitFiles(file, extension, visitor); else if (extension == null || file.getName().endsWith(extension)) visitor.visitFile(file); } } public static String extractBaseUrl(String url) { if (url == null) return null; else if (url.contains("/")) return url.substring(0, url.lastIndexOf("/")); else return url; } public static String listCanonicalUrls(Set keys) { return keys.toString(); } public static boolean isValidId(String id) { return id.matches("[A-Za-z0-9\\-\\.]{1,64}"); } public static class CaseInsensitiveSorter implements Comparator { @Override public int compare(String o1, String o2) { return o1.compareToIgnoreCase(o2); } } public static List sortedCaseInsensitive(Collection set) { List list = new ArrayList<>(); list.addAll(set); Collections.sort(list, new CaseInsensitiveSorter()); return list; } public static List sorted(Collection set) { List list = new ArrayList<>(); list.addAll(set); Collections.sort(list); return list; } public static List sorted(String[] set) { List list = new ArrayList<>(); for (String s : set) { list.add(s); } Collections.sort(list); return list; } public static List reverseSorted(Collection set) { List list = new ArrayList<>(); list.addAll(set); Collections.sort(list, Collections.reverseOrder()); return list; } public static List reverseSorted(String[] set) { List list = new ArrayList<>(); for (String s : set) { list.add(s); } Collections.sort(list, Collections.reverseOrder()); return list; } public static void analyseStringDiffs(Set source, Set target, Set missed, Set extra) { for (String s : source) if (!target.contains(s)) missed.add(s); for (String s : target) if (!source.contains(s)) extra.add(s); } /** * Only handles simple FHIRPath expressions of the type produced by the validator * * @param path * @return */ public static String fhirPathToXPath(String path) { String[] p = path.split("\\."); CommaSeparatedStringBuilder b = new CommaSeparatedStringBuilder("."); int i = 0; while (i < p.length) { String s = p[i]; if (s.contains("[")) { String si = s.substring(s.indexOf("[") + 1, s.length() - 1); if (!Utilities.isInteger(si)) throw new FHIRException("The FHIRPath expression '" + path + "' is not valid"); s = s.substring(0, s.indexOf("[")) + "[" + Integer.toString(Integer.parseInt(si) + 1) + "]"; } if (i < p.length - 1 && p[i + 1].startsWith(".ofType(")) { i++; s = s + capitalize(p[i].substring(8, p.length - 1)); } b.append(s); i++; } return b.toString(); } public static String describeDuration(Duration d) { if (d.toDays() > 2) { return String.format("%s days", d.toDays()); } else if (d.toHours() > 2) { return String.format("%s hours", d.toHours()); } else if (d.toMinutes() > 2) { return String.format("%s mins", d.toMinutes()); } else { return String.format("%s ms", d.toMillis()); } } public static String describeDuration(long ms) { long days = ms / (1000 * 60 * 60 * 24); long hours = ms / (1000 * 60 * 60) % 24; long mins = ms / (1000 * 60) % 60; long secs = ms / (1000) % 60; ms = ms % 1000; if (days > 0) { return ""+days+"d "+pad(hours,2)+":"+pad(mins,2)+":"+pad(secs,2)+"."+ms; } else { return ""+pad(hours, 2)+":"+pad(mins,2)+":"+pad(secs,2)+"."+ms; } } private static String pad(long v, int i) { return padLeft(Long.toString(v), '0', i); } public static boolean startsWithInList(String s, String... list) { if (s == null) { return false; } for (String l : list) { if (l != null) { if (s.startsWith(l)) { return true; } } } return false; } public static boolean startsWithInList(String s, Collection list) { if (s == null || list == null) { return false; } for (String l : list) { if (s.startsWith(l)) { return true; } } return false; } public static boolean endsWithInList(String s, String... list) { if (s == null) { return false; } for (String l : list) { if (s.endsWith(l)) { return true; } } return false; } public static boolean endsWithInList(String s, Collection list) { if (s == null) { return false; } for (String l : list) { if (s.endsWith(l)) { return true; } } return false; } public static final int ONE_MB = 1024; public static final String GB = "Gb"; public static final String MB = "Mb"; public static final String KB = "Kb"; public static final String BT = "b"; public static String describeSize(int length) { if (length < 0) throw new IllegalArgumentException("File length of < 0 passed in..."); if (length > Math.pow(ONE_MB, 3)) { return length / ((long) Math.pow(ONE_MB, 3)) + GB; } if (length > Math.pow(ONE_MB, 2)) { return length / ((long) Math.pow(ONE_MB, 2)) + MB; } if (length > ONE_MB) { return length / (ONE_MB) + KB; } return length + BT; } public static String describeSize(long length) { if (length < 0) throw new IllegalArgumentException("File length of < 0 passed in..."); if (length > Math.pow(ONE_MB, 3)) { return length / ((long) Math.pow(ONE_MB, 3)) + GB; } if (length > Math.pow(ONE_MB, 2)) { return length / ((long) Math.pow(ONE_MB, 2)) + MB; } if (length > ONE_MB) { return length / (ONE_MB) + KB; } return length + BT; } public static List splitBytes(byte[] array, byte[] delimiter) { List byteArrays = new LinkedList(); if (delimiter.length == 0) { return byteArrays; } int begin = 0; outer: for (int i = 0; i < array.length - delimiter.length + 1; i++) { for (int j = 0; j < delimiter.length; j++) { if (array[i + j] != delimiter[j]) { continue outer; } } // If delimiter is at the beginning then there will not be any data. if (begin < i) byteArrays.add(Arrays.copyOfRange(array, begin, i)); begin = i + delimiter.length; } // delimiter at the very end with no data following? if (begin != array.length) byteArrays.add(Arrays.copyOfRange(array, begin, array.length)); return byteArrays; } public static void unzip(InputStream zip, String target) throws IOException { unzip(zip, Path.of(target)); } public static void unzip(InputStream zip, Path target) throws IOException { try (ZipInputStream zis = new ZipInputStream(zip)) { ZipEntry zipEntry = zis.getNextEntry(); while (zipEntry != null) { boolean isDirectory = false; String n = makeOSSafe(zipEntry.getName()); if (n.endsWith(File.separator)) { isDirectory = true; } Path newPath = zipSlipProtect(n, target); if (isDirectory) { Files.createDirectories(newPath); } else { if (newPath.getParent() != null) { if (Files.notExists(newPath.getParent())) { Files.createDirectories(newPath.getParent()); } } Files.copy(zis, newPath, StandardCopyOption.REPLACE_EXISTING); } zipEntry = zis.getNextEntry(); } zis.closeEntry(); } } public static String makeOSSafe(String name) { return name.replace("\\", File.separator).replace("/", File.separator); } public static Path zipSlipProtect(String zipName, Path targetDir) throws IOException { // test zip slip vulnerability // Path targetDirResolved = targetDir.resolve("../../" + zipEntry.getName()); Path targetDirResolved = targetDir.resolve(zipName); // make sure normalized file still has targetDir as its prefix // else throws exception Path normalizePath = targetDirResolved.normalize(); if (!normalizePath.startsWith(targetDir)) { throw new IOException("Bad zip entry: " + zipName); } return normalizePath; } final static int[] illegalChars = {34, 60, 62, 124, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 58, 42, 63, 92, 47}; static { Arrays.sort(illegalChars); } public static String cleanFileName(String badFileName) { StringBuilder cleanName = new StringBuilder(); int len = badFileName.codePointCount(0, badFileName.length()); for (int i=0; i 0) { s = nv + padLeft("", '0', d); } else { int l = v.length(); int ld = l+d; if (dp > -1) { ld--; } if (nv.charAt(ld) >= '5' && !down) { s = nv.substring(0, ld-1)+((char) (nv.charAt(ld-1)+1)); } else { s = nv.substring(0, ld); } } if (s.endsWith(".")) { s = s.substring(0, s.length()-1); } return dp == -1 || dp >= s.length() ? s : s.substring(0, dp)+"."+s.substring(dp); } private static String minusOne(String value) { StringBuffer s = new StringBuffer(value); for (int i = s.length()-1; i >= 0; i--) { if (s.charAt(i) == '0') { s.setCharAt(i, '9'); } else if (s.charAt(i) != '.') { s.setCharAt(i, (char) (s.charAt(i)-1)); break; } } return s.toString(); } public static String lowBoundaryForDate(String value, int precision) { String[] res = splitTimezone(value); StringBuilder b = new StringBuilder(res[0]); if (b.length() == 4) { b.append("-01"); } if (b.length() == 7) { b.append("-01"); } if (b.length() == 10) { b.append("T00:00"); } if (b.length() == 16) { b.append(":00"); } if (b.length() == 19) { b.append(".000"); } String tz; if (precision <= 10) { tz = ""; } else { tz = Utilities.noString(res[1]) ? defLowTimezone(b.toString()) : res[1]; } return applyDatePrecision(b.toString(), precision)+tz; } private static String defLowTimezone(String string) { return "+14:00"; // Kiribati permanent timezone since 1994 and we don't worry about before then } private static String defHighTimezone(String string) { return "-12:00"; // Parts of Russia, Antarctica, Baker Island and Midway Atoll } public static String lowBoundaryForTime(String value, int precision) { String[] res = splitTimezone(value); StringBuilder b = new StringBuilder(res[0]); if (b.length() == 2) { b.append(":00"); } if (b.length() == 5) { b.append(":00"); } if (b.length() == 8) { b.append(".000"); } return applyTimePrecision(b.toString(), precision)+res[1]; } public static String highBoundaryForTime(String value, int precision) { String[] res = splitTimezone(value); StringBuilder b = new StringBuilder(res[0]); if (b.length() == 2) { b.append(":59"); } if (b.length() == 5) { b.append(":59"); } if (b.length() == 8) { b.append(".999"); } return applyTimePrecision(b.toString(), precision)+res[1]; } private static Object applyDatePrecision(String v, int precision) { switch (precision) { case 4: return v.substring(0, 4); case 6: case 7: return v.substring(0, 7); case 8: case 10: return v.substring(0, 10); case 14: return v.substring(0, 17); case 17: return v; } throw new FHIRException("Unsupported Date precision for boundary operation: "+precision); } private static Object applyTimePrecision(String v, int precision) { switch (precision) { case 2: return v.substring(0, 3); case 4: return v.substring(0, 6); case 6: return v.substring(0, 9); case 9: return v; } throw new FHIRException("Unsupported Time precision for boundary operation: "+precision); } public static String highBoundaryForDecimal(String value, int precision) { if (Utilities.noString(value)) { throw new FHIRException("Unable to calculate highBoundary for a null decimal string"); } String e = value.contains("e") ? value.substring(value.indexOf("e")+1) : null; if (value.contains("e")) { value = value.substring(0, value.indexOf("e")); } if (isZero(value)) { return applyPrecision("0.50000000000000000000000000000", precision, false); } else if (value.startsWith("-")) { return "-"+lowBoundaryForDecimal(value.substring(1), precision)+(e == null ? "" : e); } else { if (value.contains(".")) { return applyPrecision(value+"50000000000000000000000000000", precision, false)+(e == null ? "" : e); } else { return applyPrecision(value+".50000000000000000000000000000", precision, false)+(e == null ? "" : e); } } } private static boolean isZero(String value) { return value.replace(".", "").replace("-", "").replace("0", "").length() == 0; } public static String highBoundaryForDate(String value, int precision) { String[] res = splitTimezone(value); StringBuilder b = new StringBuilder(res[0]); if (b.length() == 4) { b.append("-12"); } if (b.length() == 7) { b.append("-"+dayCount(Integer.parseInt(b.substring(0,4)), Integer.parseInt(b.substring(5,7)))); } if (b.length() == 10) { b.append("T23:59"); } if (b.length() == 16) { b.append(":59"); } if (b.length() == 19) { b.append(".999"); } String tz; if (precision <= 10) { tz = ""; } else { tz = Utilities.noString(res[1]) ? defHighTimezone(b.toString()) : res[1]; } return applyDatePrecision(b.toString(), precision)+tz; } private static String dayCount(int y, int m) { switch (m) { case 1: return "31"; case 2: return ((y % 4 == 0) && (y % 400 == 0 || !(y % 100 == 0))) ? "29" : "28"; case 3: return "31"; case 4: return "30"; case 5: return "31"; case 6: return "30"; case 7: return "31"; case 8: return "31"; case 9: return "30"; case 10: return "31"; case 11: return "30"; case 12: return "31"; default: return "30"; // make the compiler happy } } public static Integer getDecimalPrecision(String value) { if (value.contains("e")) { value = value.substring(0, value.indexOf("e")); } if (value.contains(".")) { return value.split("\\.")[1].length(); } else { return 0; } } private static String[] splitTimezone(String value) { String[] res = new String[2]; if (value.contains("+")) { res[0] = value.substring(0, value.indexOf("+")); res[1] = value.substring(value.indexOf("+")); } else if (value.contains("-") && value.contains("T") && value.lastIndexOf("-") > value.indexOf("T")) { res[0] = value.substring(0, value.lastIndexOf("-")); res[1] = value.substring(value.lastIndexOf("-")); } else if (value.contains("Z")) { res[0] = value.substring(0, value.indexOf("Z")); res[1] = value.substring(value.indexOf("Z")); } else { res[0] = value; res[1] = ""; } return res; } public static Integer getDatePrecision(String value) { return splitTimezone(value)[0].replace("-", "").replace("T", "").replace(":", "").replace(".", "").length(); } public static Integer getTimePrecision(String value) { return splitTimezone(value)[0].replace("T", "").replace(":", "").replace(".", "").length(); } public static String padInt(int i, int len) { return Utilities.padLeft(Integer.toString(i), ' ', len); } public static String padInt(long i, int len) { return Utilities.padLeft(Long.toString(i), ' ', len); } public static Object makeSingleLine(String text) { text = text.replace("\r", " "); text = text.replace("\n", " "); while (text.contains(" ")) { text = text.replace(" ", " "); } return text; } public static int parseInt(String value, int def) { if (isInteger(value)) { return Integer.parseInt(value); } else { return def; } } /** * Appends a text from a derived element to its base element. * * @param baseText The text set in the base element, or {@code null}. * @param derivedText The text set in the derived element, starting with "...". * @return The resulting text. */ public static String appendDerivedTextToBase(@Nullable final String baseText, final String derivedText) { if (baseText == null) { return derivedText.substring(3); } return baseText + "\r\n" + derivedText.substring(3); } public static void deleteEmptyFolders(File df) { for (File f : df.listFiles()) { if (f.isDirectory()) { deleteEmptyFolders(f); } } boolean empty = true; for (File f : df.listFiles()) { empty = false; break; } if (empty) { df.delete(); } } public static String getRelativePath(String root, String path) { String res = path.substring(root.length()); if (res.startsWith(File.separator)) { res = res.substring(1); } return res; } public static String getRelativeUrlPath(String root, String path) { String res = path.substring(root.length()); if (res.startsWith("/")) { res = res.substring(1); } return res; } public static List listAllFiles(String path, List ignoreList) throws IOException { List res = new ArrayList<>(); addAllFiles(res, path, ManagedFileAccess.file(path), ignoreList); return res; } private static void addAllFiles(List res, String root, File dir, List ignoreList) { for (File f : dir.listFiles()) { if (ignoreList == null || !ignoreList.contains(f.getAbsolutePath())) { if (f.isDirectory()) { addAllFiles(res, root, f, ignoreList); } else if (!f.getName().equals(".DS_Store")) { res.add(getRelativePath(root, f.getAbsolutePath())); } } } } public static boolean isValidCRName(String name) { return name != null && name.matches("[A-Z]([A-Za-z0-9_]){1,254}"); } public static boolean isAllWhitespace(String s) { if (Utilities.noString(s)) { return true; } for (char ch : s.toCharArray()) { if (!isWhitespace(ch)) { return false; } } return true; } public static String trimWS(String s) { if (Utilities.noString(s)) { return s; } int start = 0; while (start < s.length() && isWhitespace(s.charAt(start))) { start++; } if (start == s.length()) { return ""; } int end = s.length() - 1; while (end >= 0 && isWhitespace(s.charAt(end))) { end--; } if (start > end) { return ""; } return s.substring(start, end+1); } // from https://en.wikipedia.org/wiki/Whitespace_character#Unicode public static boolean isWhitespace(int ch) { return Utilities.existsInList(ch, '\u0009', '\n', '\u000B','\u000C','\r','\u0020','\u0085','\u00A0', '\u1680','\u2000','\u2001','\u2002','\u2003','\u2004','\u2005','\u2006','\u2007','\u2008','\u2009','\u200A', '\u2028', '\u2029', '\u202F', '\u205F', '\u3000'); } public static boolean stringsEqual(String s1, String s2) { if (s1 == null && s2 == null) { return true; } else if (s1 == null) { return false; } else { return s1.equals(s2); } } public static String tail(String url) { int i = url.length()-1; while (i >= 0 && (isTokenChar(url.charAt(i)) || isDigit(url.charAt(i))) ) { i--; } if (i < 0) { return url; } else { return url.substring(i+1); } } public static List strings(String... members) { List ret = new ArrayList<>(); for (String m : members) { ret.add(m); } return ret; } public static List splitStrings(String src, String regex) { List ret = new ArrayList<>(); for (String m : src.split(regex)) { ret.add(m); } return ret; } public static String stripPara(String p) { if (noString(p)) { return ""; } p = p.trim(); if (p.startsWith("

")) { p = p.substring(3); } if (p.endsWith("

")) { p = p.substring(0, p.length()-4); } return p; } public static String stripAllPara(String p) { if (noString(p)) { return ""; } p = p.trim(); if (p.startsWith("

")) { p = p.substring(3); } if (p.endsWith("

")) { p = p.substring(0, p.length()-4); } p = p.replace("

", " "); p = p.replace("

", ""); while (p.contains("

0 && Character.isWhitespace(s.charAt(i))) { i--; } return i == 0 ? "" : s.substring(0, i+1); } public static void renameDirectory(String source, String dest) throws FHIRException, IOException { File src = ManagedFileAccess.file(source); File dst = ManagedFileAccess.file(dest); if (!src.renameTo(dst)) { int i = 0; do { try { Thread.sleep(20); } catch (Exception e) { // nothing } System.gc(); i++; } while (!src.renameTo(dst) && i < 10); if (src.exists()) { copyDirectory(source, dest, null); try { src.delete(); } catch (Exception e) { // nothing } } } } public static String urlTail(String url) { if (url == null) { return null; } return url.contains("/") ? url.substring(url.lastIndexOf("/")+1) : url; } public static String escapeSql(String s) { return s.replace("'", "''"); } public static String[] simpleSplit(String cnt, String div) { if (cnt == null) { return new String[] {}; } List parts = new ArrayList<>(); int cursor = 0; int last = 0; while (cursor < cnt.length()) { if (matches(cnt, div, cursor)) { parts.add(cnt.substring(last, cursor)); cursor = cursor + div.length(); last = cursor; } else { cursor++; } } parts.add(cnt.substring(last, cursor)); return parts.toArray(new String[] {}); } private static boolean matches(String cnt, String div, int cursor) { if (div.length() + cursor > cnt.length()) { return false; } for (int i = 0; i < div.length(); i++) { if (cnt.charAt(cursor+i) != div.charAt(i)) { return false; } } return true; } public static String stripEoln(String text) { if (text == null) { return ""; } return text.replace("\r\n", " ").replace("\n", " ").replace("\r", " "); } public static String extractDomain(String source) { try { URI uri = URI.create(source); return uri.getHost(); } catch (Exception e) { return "??"; } } public static boolean isValidHtmlAnchorChar(char c) { if ( (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9')) { return true; } switch (c) { case '!': case '$': case '&': case '\'': case '(': case ')': case '*': case '+': case ',': case ';': case '=': case '.': case '_': case '-': case '~': case ':': case '@': case '/': case '?': return true; default: return false; } } public static boolean listValueStartsWith(String s, Set list) { if (s == null || list == null) { return false; } for (String l : list) { if (l.startsWith(s)) { return true; } } return false; } public static String extractByRegex(String input, String regex) { Pattern pattern = Pattern.compile(regex); Matcher matcher = pattern.matcher(input); StringBuilder result = new StringBuilder(); while (matcher.find()) { result.append(matcher.group(1)); } return result.length() == 0 ? null : result.toString(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy