/* * Copyright (C)2005-2018 Haxe Foundation * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. */ import php.*; @:coreApi class StringTools { public inline static function urlEncode( s : String ) : String { return Global.rawurlencode(s); } public inline static function urlDecode( s : String ) : String { return Global.urldecode(s); } public inline static function htmlEscape( s : String, ?quotes : Bool ) : String { return Global.htmlspecialchars(s, (quotes ? Const.ENT_QUOTES | Const.ENT_HTML401 : Const.ENT_NOQUOTES)); } public inline static function htmlUnescape( s : String ) : String { return Global.htmlspecialchars_decode(s, Const.ENT_QUOTES); } public static function startsWith( s : String, start : String ) : Bool { return start == '' || Global.strpos(s, start) == 0; } public static function endsWith( s : String, end : String ) : Bool { return end == '' || Global.substr(s, -end.length) == end; } public static function isSpace( s : String, pos : Int ) : Bool { var c = s.charCodeAt( pos ); return (c >= 9 && c <= 13) || c == 32; } public inline static function ltrim( s : String ) : String { return Global.ltrim(s); } public inline static function rtrim( s : String ) : String { return Global.rtrim(s); } public inline static function trim( s : String ) : String { return Global.trim(s); } public static function rpad( s : String, c : String, l : Int ) : String { if (c.length == 0 || s.length >= l) return s; var padLength = Math.ceil((l - s.length) / c.length) * c.length + s.length; return Global.str_pad(s, padLength, c, Const.STR_PAD_RIGHT); } public static function lpad( s : String, c : String, l : Int ) : String { if (c.length == 0 || s.length >= l) return s; var padLength = Math.ceil((l - s.length) / c.length) * c.length + s.length; return Global.str_pad(s, padLength, c, Const.STR_PAD_LEFT); } public static function replace( s : String, sub : String, by : String ) : String { if (sub == '') { return Global.implode(by, Global.str_split(s)); } return Global.str_replace(sub, by, s); } public static function hex( n : Int, ?digits : Int ) : String { var s = Global.dechex(n); var len = 8; if (s.length > (null == digits ? len : (len = digits > len ? digits : len))) s = s.substr(-len); else if ( digits != null ) s = lpad(s, '0', digits); return s.toUpperCase(); } public static inline function fastCodeAt( s : String, index : Int ) : Int { return (s.length == index ? 0 : Global.ord((s:NativeString)[index])); } public static inline function isEof( c : Int ) : Bool { return c == 0; } /** Returns a String that can be used as a single command line argument on Unix. The input will be quoted, or escaped if necessary. */ public static function quoteUnixArg(argument:String):String { // Based on cpython's shlex.quote(). // https://hg.python.org/cpython/file/a3f076d4f54f/Lib/shlex.py#l278 if (argument == "") return "''"; if (!~/[^a-zA-Z0-9_@%+=:,.\/-]/.match(argument)) return argument; // use single quotes, and put single quotes into double quotes // the string $'b is then quoted as '$'"'"'b' return "'" + replace(argument, "'", "'\"'\"'") + "'"; } /** Character codes of the characters that will be escaped by `quoteWinArg(_, true)`. */ public static var winMetaCharacters = [";".code, ",".code, " ".code, "(".code, ")".code, "%".code, "!".code, "^".code, "\"".code, "<".code, ">".code, "&".code, "|".code, "\n".code, "\r".code]; /** Returns a String that can be used as a single command line argument on Windows. The input will be quoted, or escaped if necessary, such that the output will be parsed as a single argument using the rule specified in http://msdn.microsoft.com/en-us/library/ms880421 Examples: ``` quoteWinArg("abc") == "abc"; quoteWinArg("ab c") == '"ab c"'; ``` */ public static function quoteWinArg(argument:String, escapeMetaCharacters:Bool):String { // If there is no space, tab, back-slash, or double-quotes, and it is not an empty string. if (!~/^[^ \t\\"]+$/.match(argument)) { // Based on cpython's subprocess.list2cmdline(). // https://hg.python.org/cpython/file/50741316dd3a/Lib/subprocess.py#l620 var result = new StringBuf(); var needquote = argument.indexOf(" ") != -1 || argument.indexOf("\t") != -1 || argument == ""; if (needquote) result.add('"'); var bs_buf = new StringBuf(); for (i in 0...argument.length) { switch (argument.charCodeAt(i)) { case "\\".code: // Don't know if we need to double yet. bs_buf.add("\\"); case '"'.code: // Double backslashes. var bs = bs_buf.toString(); result.add(bs); result.add(bs); bs_buf = new StringBuf(); result.add('\\"'); case var c: // Normal char if (bs_buf.length > 0) { result.add(bs_buf.toString()); bs_buf = new StringBuf(); } result.addChar(c); } } // Add remaining backslashes, if any. result.add(bs_buf.toString()); if (needquote) { result.add(bs_buf.toString()); result.add('"'); } argument = result.toString(); } if (escapeMetaCharacters) { var result = new StringBuf(); for (i in 0...argument.length) { var c = argument.charCodeAt(i); if (winMetaCharacters.indexOf(c) >= 0) { result.addChar("^".code); } result.addChar(c); } return result.toString(); } else { return argument; } } }