From 2e271dd6fcc4e2e590cbb6813e702b789683389e Mon Sep 17 00:00:00 2001 From: veryben Date: Thu, 18 May 2017 17:16:55 +0800 Subject: [PATCH 1/4] =?UTF-8?q?=E6=B7=BB=E5=8A=A0.gitignore=E6=96=87?= =?UTF-8?q?=E4=BB=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 14 ++++++++++++++ 1 file changed, 14 insertions(+) create mode 100644 .gitignore diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..789c91d --- /dev/null +++ b/.gitignore @@ -0,0 +1,14 @@ +bin +build +out +target +.settings +.classpath +.project +.gradle +.DS_Store +*.iml +*.ipr +*.iws +*.log +.idea From ec1ed97ae547acc1d2a6cb2b8f59c8fec254cce5 Mon Sep 17 00:00:00 2001 From: veryben Date: Thu, 18 May 2017 17:36:43 +0800 Subject: [PATCH 2/4] =?UTF-8?q?=E4=BF=AE=E5=A4=8DIniFileReader=E8=AF=BB?= =?UTF-8?q?=E5=8F=96=E5=90=8C=E5=90=8D=E5=AD=97=E6=AE=B5(tracker=5Fserver)?= =?UTF-8?q?=E5=8F=AA=E8=83=BD=E5=8F=96=E6=9C=80=E5=90=8E=E4=B8=80=E4=B8=AA?= =?UTF-8?q?=E7=9A=84bug?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/org/csource/common/IniFileReader.java | 152 ++++++++++-------- .../fastdfs/test/IniFileReaderTests.java | 29 ++++ 2 files changed, 117 insertions(+), 64 deletions(-) create mode 100644 src/org/csource/fastdfs/test/IniFileReaderTests.java diff --git a/src/org/csource/common/IniFileReader.java b/src/org/csource/common/IniFileReader.java index 727e100..6be87e4 100644 --- a/src/org/csource/common/IniFileReader.java +++ b/src/org/csource/common/IniFileReader.java @@ -21,7 +21,7 @@ public class IniFileReader { private Hashtable paramTable; private String conf_filename; - + /** * @param conf_filename config filename */ @@ -30,7 +30,7 @@ public class IniFileReader this.conf_filename = conf_filename; loadFromFile(conf_filename); } - + /** * get the config filename * @return config filename @@ -39,7 +39,7 @@ public class IniFileReader { return this.conf_filename; } - + /** * get string value from config file * @param name item name in config file @@ -53,12 +53,12 @@ public class IniFileReader { return null; } - + if (obj instanceof String) { return (String)obj; } - + return (String)((ArrayList)obj).get(0); } @@ -75,7 +75,7 @@ public class IniFileReader { return default_value; } - + return Integer.parseInt(szValue); } @@ -92,11 +92,11 @@ public class IniFileReader { return default_value; } - - return szValue.equalsIgnoreCase("yes") || szValue.equalsIgnoreCase("on") || + + return szValue.equalsIgnoreCase("yes") || szValue.equalsIgnoreCase("on") || szValue.equalsIgnoreCase("true") || szValue.equals("1"); } - + /** * get all values from config file * @param name item name in config file @@ -106,77 +106,101 @@ public class IniFileReader { Object obj; String[] values; - + obj = this.paramTable.get(name); if (obj == null) { return null; } - + if (obj instanceof String) { values = new String[1]; values[0] = (String)obj; return values; } - + Object[] objs = ((ArrayList)obj).toArray(); values = new String[objs.length]; System.arraycopy(objs, 0, values, 0, objs.length); return values; } - - private void loadFromFile(String conf_filename) throws FileNotFoundException, IOException - { - //问题说明 使用中发现原来客户端打jar包后,在另一个项目中引用,另一个项目打jar包后运行时找不到客户端配置文件 - String name; - String value; - Object obj; - ArrayList valueList; - InputStream is=null; - this.paramTable = new Hashtable(); - try - { - Properties props; - try { - is = Thread.currentThread().getContextClassLoader().getResourceAsStream(conf_filename); - props = new Properties(); - props.load(is); - } catch (Exception ex) { - is = new FileInputStream(conf_filename); - props = new Properties(); - props.load(is); - } - Iterator> it = props.entrySet().iterator(); - while (it.hasNext()) { - Map.Entry entry = it.next(); - name= entry.getKey().toString(); - value = entry.getValue().toString(); - obj = this.paramTable.get(name); - if (obj == null) - { - this.paramTable.put(name, value); - } - else if (obj instanceof String) - { - valueList = new ArrayList(); - valueList.add(obj); - valueList.add(value); - this.paramTable.put(name, valueList); - } - else - { - valueList = (ArrayList)obj; - valueList.add(value); - } - } - } - finally - { - if (is != null) { - is.close(); - } - } + private void loadFromFile(String confFilePath) throws IOException { + InputStream in = null; + try { + // 从类路径加载 + in = classLoader().getResourceAsStream(confFilePath); + //System.out.println("loadFrom...class path done"); + readToParamTable(in); + } catch (Exception ex) { + ex.printStackTrace(); + } finally { + try { + if(in != null) in.close(); + //System.out.println("loadFrom...finally...in.close(); done"); + } catch (Exception ex) { + ex.printStackTrace(); + } } + } + + private void readToParamTable(InputStream in) throws IOException { + this.paramTable = new Hashtable(); + String line; + String[] parts; + String name; + String value; + Object obj; + ArrayList valueList; + InputStreamReader inReader = null; + BufferedReader bufferedReader = null; + try { + inReader = new InputStreamReader(in); + bufferedReader = new BufferedReader(inReader); + while ((line = bufferedReader.readLine()) != null) { + line = line.trim(); + if (line.length() == 0 || line.charAt(0) == '#') { + continue; + } + parts = line.split("=", 2); + if (parts.length != 2) { + continue; + } + name = parts[0].trim(); + value = parts[1].trim(); + obj = this.paramTable.get(name); + if (obj == null) { + this.paramTable.put(name, value); + } else if (obj instanceof String) { + valueList = new ArrayList(); + valueList.add(obj); + valueList.add(value); + this.paramTable.put(name, valueList); + } else { + valueList = (ArrayList) obj; + valueList.add(value); + } + } + } catch (Exception ex) { + ex.printStackTrace(); + } finally { + try { + if(bufferedReader != null) bufferedReader.close(); + if(inReader != null) inReader.close(); + //System.out.println("readToParamTable...finally...bufferedReader.close();inReader.close(); done"); + } catch (Exception ex) { + ex.printStackTrace(); + } + } + } + + private static ClassLoader classLoader() { + ClassLoader loader = Thread.currentThread().getContextClassLoader(); + if (loader == null) { + loader = ClassLoader.getSystemClassLoader(); + } + return loader; + } + } diff --git a/src/org/csource/fastdfs/test/IniFileReaderTests.java b/src/org/csource/fastdfs/test/IniFileReaderTests.java new file mode 100644 index 0000000..4a95c71 --- /dev/null +++ b/src/org/csource/fastdfs/test/IniFileReaderTests.java @@ -0,0 +1,29 @@ +package org.csource.fastdfs.test; + +import org.csource.common.IniFileReader; + +/** + * Created by James on 2017/5/16. + */ +public class IniFileReaderTests { + + public static void main(String[] args) throws Exception { + String conf_filename = "fdfs_client.conf"; + IniFileReader iniFileReader = new IniFileReader(conf_filename); + System.out.println("getConfFilename: " + iniFileReader.getConfFilename()); + System.out.println("connect_timeout: " + iniFileReader.getIntValue("connect_timeout", 3)); + System.out.println("network_timeout: " + iniFileReader.getIntValue("network_timeout", 45)); + System.out.println("charset: " + iniFileReader.getStrValue("charset")); + System.out.println("http.tracker_http_port: " + iniFileReader.getIntValue("http.tracker_http_port", 8080)); + System.out.println("http.anti_steal_token: " + iniFileReader.getBoolValue("http.anti_steal_token", false)); + System.out.println("http.secret_key: " + iniFileReader.getStrValue("http.secret_key")); + String[] tracker_servers = iniFileReader.getValues("tracker_server"); + if(tracker_servers != null) { + System.out.println("tracker_servers.length: " + tracker_servers.length); + for (int i = 0; i < tracker_servers.length; i++) { + System.out.println(String.format("tracker_servers[%s]: %s", i, tracker_servers[i])); + } + } + } + +} From 51914bf1ac4eb661ffe4c18f196d288741b483e3 Mon Sep 17 00:00:00 2001 From: veryben Date: Thu, 18 May 2017 17:40:22 +0800 Subject: [PATCH 3/4] =?UTF-8?q?=E5=A2=9E=E5=8A=A0IniFileReader=E4=BC=98?= =?UTF-8?q?=E5=85=88=E8=AF=BB=E5=8F=96=E6=96=87=E4=BB=B6=E7=B3=BB=E7=BB=9F?= =?UTF-8?q?=E9=85=8D=E7=BD=AE=E6=96=87=E4=BB=B6=E5=8A=9F=E8=83=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/org/csource/common/IniFileReader.java | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/src/org/csource/common/IniFileReader.java b/src/org/csource/common/IniFileReader.java index 6be87e4..e872829 100644 --- a/src/org/csource/common/IniFileReader.java +++ b/src/org/csource/common/IniFileReader.java @@ -129,9 +129,16 @@ public class IniFileReader private void loadFromFile(String confFilePath) throws IOException { InputStream in = null; try { + // 优先从文件系统路径加载 + if (new File(confFilePath).exists()) { + in = new FileInputStream(confFilePath); + //System.out.println("loadFrom...file path done"); + } // 从类路径加载 - in = classLoader().getResourceAsStream(confFilePath); - //System.out.println("loadFrom...class path done"); + else { + in = classLoader().getResourceAsStream(confFilePath); + //System.out.println("loadFrom...class path done"); + } readToParamTable(in); } catch (Exception ex) { ex.printStackTrace(); From 69417006f4534a1fb4a7e2e0ecf6528e4eeece4d Mon Sep 17 00:00:00 2001 From: veryben Date: Thu, 18 May 2017 20:42:57 +0800 Subject: [PATCH 4/4] =?UTF-8?q?=E8=A7=84=E8=8C=83maven=E9=A1=B9=E7=9B=AE?= =?UTF-8?q?=E7=BB=93=E6=9E=84=E5=B9=B6=E5=8D=87=E7=BA=A7=E7=89=88=E6=9C=AC?= =?UTF-8?q?=E5=8F=B7(1.27-SNAPSHOT)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/HISTORY => HISTORY | 0 src/README => README | 10 +- build.xml | 37 + fdfs_client.conf | 10 + pom.xml | 75 +- src/build.xml | 55 - .../java}/org/csource/common/Base64.java | 1078 ++++++++--------- .../org/csource/common/IniFileReader.java | 0 .../java}/org/csource/common/MyException.java | 0 .../org/csource/common/NameValuePair.java | 0 .../org/csource/fastdfs/ClientGlobal.java | 0 .../org/csource/fastdfs/DownloadCallback.java | 0 .../org/csource/fastdfs/DownloadStream.java | 100 +- .../java}/org/csource/fastdfs/FileInfo.java | 0 .../org/csource/fastdfs/ProtoCommon.java | 0 .../csource/fastdfs/ProtoStructDecoder.java | 0 .../java}/org/csource/fastdfs/ServerInfo.java | 0 .../org/csource/fastdfs/StorageClient.java | 0 .../org/csource/fastdfs/StorageClient1.java | 0 .../org/csource/fastdfs/StorageServer.java | 0 .../java}/org/csource/fastdfs/StructBase.java | 0 .../org/csource/fastdfs/StructGroupStat.java | 0 .../csource/fastdfs/StructStorageStat.java | 0 .../org/csource/fastdfs/TrackerClient.java | 0 .../org/csource/fastdfs/TrackerGroup.java | 0 .../org/csource/fastdfs/TrackerServer.java | 0 .../org/csource/fastdfs/UploadCallback.java | 0 .../org/csource/fastdfs/UploadStream.java | 0 .../fastdfs/test/DownloadFileWriter.java | 0 .../org/csource/fastdfs/test/Monitor.java | 0 .../java}/org/csource/fastdfs/test/Test.java | 0 .../java}/org/csource/fastdfs/test/Test1.java | 0 .../csource/fastdfs/test/TestAppender.java | 0 .../csource/fastdfs/test/TestAppender1.java | 0 .../org/csource/fastdfs/test/TestClient.java | 0 .../org/csource/fastdfs/test/TestClient1.java | 0 .../org/csource/fastdfs/test/TestLoad.java | 0 .../fastdfs/test/UploadLocalFileSender.java | 0 .../resources/fdfs_client.conf.sample} | 0 .../csource/common}/IniFileReaderTests.java | 6 +- src/test/resources/fdfs_client.conf | 9 + 41 files changed, 681 insertions(+), 699 deletions(-) rename src/HISTORY => HISTORY (100%) rename src/README => README (60%) create mode 100644 build.xml create mode 100644 fdfs_client.conf delete mode 100644 src/build.xml rename src/{ => main/java}/org/csource/common/Base64.java (96%) rename src/{ => main/java}/org/csource/common/IniFileReader.java (100%) rename src/{ => main/java}/org/csource/common/MyException.java (100%) rename src/{ => main/java}/org/csource/common/NameValuePair.java (100%) rename src/{ => main/java}/org/csource/fastdfs/ClientGlobal.java (100%) rename src/{ => main/java}/org/csource/fastdfs/DownloadCallback.java (100%) rename src/{ => main/java}/org/csource/fastdfs/DownloadStream.java (95%) rename src/{ => main/java}/org/csource/fastdfs/FileInfo.java (100%) rename src/{ => main/java}/org/csource/fastdfs/ProtoCommon.java (100%) rename src/{ => main/java}/org/csource/fastdfs/ProtoStructDecoder.java (100%) rename src/{ => main/java}/org/csource/fastdfs/ServerInfo.java (100%) rename src/{ => main/java}/org/csource/fastdfs/StorageClient.java (100%) rename src/{ => main/java}/org/csource/fastdfs/StorageClient1.java (100%) rename src/{ => main/java}/org/csource/fastdfs/StorageServer.java (100%) rename src/{ => main/java}/org/csource/fastdfs/StructBase.java (100%) rename src/{ => main/java}/org/csource/fastdfs/StructGroupStat.java (100%) rename src/{ => main/java}/org/csource/fastdfs/StructStorageStat.java (100%) rename src/{ => main/java}/org/csource/fastdfs/TrackerClient.java (100%) rename src/{ => main/java}/org/csource/fastdfs/TrackerGroup.java (100%) rename src/{ => main/java}/org/csource/fastdfs/TrackerServer.java (100%) rename src/{ => main/java}/org/csource/fastdfs/UploadCallback.java (100%) rename src/{ => main/java}/org/csource/fastdfs/UploadStream.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/DownloadFileWriter.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/Monitor.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/Test.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/Test1.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/TestAppender.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/TestAppender1.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/TestClient.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/TestClient1.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/TestLoad.java (100%) rename src/{ => main/java}/org/csource/fastdfs/test/UploadLocalFileSender.java (100%) rename src/{fdfs_client.conf => main/resources/fdfs_client.conf.sample} (100%) rename src/{org/csource/fastdfs/test => test/java/org/csource/common}/IniFileReaderTests.java (91%) create mode 100644 src/test/resources/fdfs_client.conf diff --git a/src/HISTORY b/HISTORY similarity index 100% rename from src/HISTORY rename to HISTORY diff --git a/src/README b/README similarity index 60% rename from src/README rename to README index fa8c6a3..20d4b1e 100644 --- a/src/README +++ b/README @@ -1,27 +1,27 @@ Copyright (C) 2008 Happy Fish / YuQing -FastDFS Java Client API may be copied only under the terms of +FastDFS Java Client API may be copied only under the terms of the BSD license. Please visit the FastDFS Home Page for more detail. English language: http://english.csource.org/ Chinese language: http://www.csource.org/ -The jar file is compiled by JDK1.5, you can download the last version +The jar file is compiled by JDK1.5, you can download the last version from google code: http://code.google.com/p/fastdfs/downloads/list run the FastDFS Java Client test: java -cp org.csource.fastdfs.test.TestClient eg.: -java -cp fastdfs_client_v1.25.jar org.csource.fastdfs.test.TestClient fdfs_client.conf c:\windows\system32\notepad.exe +java -cp org.csource.fastdfs.test.TestClient fdfs_client.conf c:\windows\system32\notepad.exe or: -java -cp fastdfs_client_v1.25.jar org.csource.fastdfs.test.TestClient fdfs_client.conf /usr/include/stdlib.h +java -cp org.csource.fastdfs.test.TestClient fdfs_client.conf /usr/include/stdlib.h run the FastDFS monitor: java -cp org.csource.fastdfs.test.Monitor eg.: -java -cp fastdfs_client_v1.25.jar org.csource.fastdfs.test.Monitor fdfs_client.conf +java -cp org.csource.fastdfs.test.Monitor fdfs_client.conf diff --git a/build.xml b/build.xml new file mode 100644 index 0000000..29a2337 --- /dev/null +++ b/build.xml @@ -0,0 +1,37 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/fdfs_client.conf b/fdfs_client.conf new file mode 100644 index 0000000..f28fd40 --- /dev/null +++ b/fdfs_client.conf @@ -0,0 +1,10 @@ +connect_timeout = 2 +network_timeout = 30 +charset = UTF-8 +http.tracker_http_port = 8080 +http.anti_steal_token = no +http.secret_key = FastDFS1234567890 + +tracker_server = 10.0.11.247:22122 +tracker_server = 10.0.11.248:22122 +tracker_server = 10.0.11.249:22122 diff --git a/pom.xml b/pom.xml index a64f34c..c29c3bd 100644 --- a/pom.xml +++ b/pom.xml @@ -1,52 +1,35 @@ - 4.0.0 + 4.0.0 - org.csource - fastdfs-client-java - 1.26 - fastdfs-client-java - fastdfs client with java - jar + org.csource + fastdfs-client-java + 1.27-SNAPSHOT + fastdfs-client-java + fastdfs client for java + jar - - UTF-8 - UTF-8 - UTF-8 - + + UTF-8 + UTF-8 + true + true + 1.5 + + + + + + org.apache.maven.plugins + maven-compiler-plugin + 3.5.1 + + UTF-8 + ${jdk.version} + ${jdk.version} + + + + - - src - - - org.apache.maven.plugins - maven-compiler-plugin - 2.5.1 - - UTF-8 - true - 1.5 - 1.5 - - - - org.apache.maven.plugins - maven-jar-plugin - 2.4 - - - package - - jar - - - - **/test/*.class - - - - - - - diff --git a/src/build.xml b/src/build.xml deleted file mode 100644 index 44693f1..0000000 --- a/src/build.xml +++ /dev/null @@ -1,55 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/src/org/csource/common/Base64.java b/src/main/java/org/csource/common/Base64.java similarity index 96% rename from src/org/csource/common/Base64.java rename to src/main/java/org/csource/common/Base64.java index f044b0f..3549d22 100644 --- a/src/org/csource/common/Base64.java +++ b/src/main/java/org/csource/common/Base64.java @@ -1,539 +1,539 @@ -package org.csource.common; - -import java.io.IOException; - -/** - * Freeware from: - * Roedy Green - * Canadian Mind Products - * #327 - 964 Heywood Avenue - * Victoria, BC Canada V8V 2Y5 - * tel:(250) 361-9093 - * mailto:roedy@mindprod.com - */ - -/** - * Encode arbitrary binary into printable ASCII using BASE64 encoding. - * very loosely based on the Base64 Reader by - * Dr. Mark Thornton - * Optrak Distribution Software Ltd. - * http://www.optrak.co.uk - * and Kevin Kelley's http://www.ruralnet.net/~kelley/java/Base64.java - * - * Base64 is a way of encoding 8-bit characters using only ASCII printable - * characters similar to UUENCODE. UUENCODE includes a filename where BASE64 does not. - * The spec is described in RFC 2045. Base64 is a scheme where - * 3 bytes are concatenated, then split to form 4 groups of 6-bits each; and - * each 6-bits gets translated to an encoded printable ASCII character, via a - * table lookup. An encoded string is therefore longer than the original by - * about 1/3. The "=" character is used to pad the end. Base64 is used, - * among other things, to encode the user:password string in an - * Authorization: header for HTTP. Don't confuse Base64 with - * x-www-form-urlencoded which is handled by - * Java.net.URLEncoder.encode/decode - * If you don't like this code, there is another implementation at http://www.ruffboy.com/download.htm - * Sun has an undocumented method called sun.misc.Base64Encoder.encode. - * You could use hex, simpler to code, but not as compact. - * - * If you wanted to encode a giant file, you could do it in large chunks that - * are even multiples of 3 bytes, except for the last chunk, and append the outputs. - * - * To encode a string, rather than binary data java.net.URLEncoder may be better. See - * printable characters in the Java glossary for a discussion of the differences. - * - * version 1.4 2002 February 15 -- correct bugs with uneven line lengths, - * allow you to configure line separator. - * now need Base64 object and instance methods. - * new mailing address. - * version 1.3 2000 September 12 -- fix problems with estimating output length in encode - * version 1.2 2000 September 09 -- now handles decode as well. - * version 1.1 1999 December 04 -- more symmetrical encoding algorithm. - * more accurate StringBuffer allocation size. - * version 1.0 1999 December 03 -- posted in comp.lang.java.programmer. - * Futures Streams or files. - */ - -public class Base64 -{ - - /** - * how we separate lines, e.g. \n, \r\n, \r etc. - */ - private String lineSeparator = System.getProperty( "line.separator" ); - - /** - * max chars per line, excluding lineSeparator. A multiple of 4. - */ - private int lineLength = 72; - - private char[] valueToChar = new char[64]; - - /** - * binary value encoded by a given letter of the alphabet 0..63 - */ - private int[] charToValue = new int[256]; - - private int[] charToPad = new int[4]; - - /* constructor */ - public Base64() - { - this.init('+', '/', '='); - } - - /* constructor */ - public Base64(char chPlus, char chSplash, char chPad, int lineLength) - { - this.init(chPlus, chSplash, chPad); - this.lineLength = lineLength; - } - - public Base64(int lineLength) - { - this.lineLength = lineLength; - } - - /* initialise defaultValueToChar and defaultCharToValue tables */ - private void init(char chPlus, char chSplash, char chPad) - { - int index = 0; - // build translate this.valueToChar table only once. - // 0..25 -> 'A'..'Z' - for ( int i='A'; i<='Z'; i++) { - this.valueToChar[index++] = (char)i; - } - - // 26..51 -> 'a'..'z' - for ( int i='a'; i<='z'; i++ ) { - this.valueToChar[index++] = (char)i; - } - - // 52..61 -> '0'..'9' - for ( int i='0'; i<='9'; i++) { - this.valueToChar[index++] = (char)i; - } - - this.valueToChar[index++] = chPlus; - this.valueToChar[index++] = chSplash; - - // build translate defaultCharToValue table only once. - for ( int i=0; i<256; i++ ) - { - this.charToValue[i] = IGNORE; // default is to ignore - } - - for ( int i=0; i<64; i++ ) - { - this.charToValue[this.valueToChar[i]] = i; - } - - this.charToValue[chPad] = PAD; - java.util.Arrays.fill(this.charToPad, chPad); - } - - /** - * Encode an arbitrary array of bytes as Base64 printable ASCII. - * It will be broken into lines of 72 chars each. The last line is not - * terminated with a line separator. - * The output will always have an even multiple of data characters, - * exclusive of \n. It is padded out with =. - */ - public String encode(byte[] b) throws IOException - { - // Each group or partial group of 3 bytes becomes four chars - // covered quotient - int outputLength = ((b.length + 2) / 3) * 4; - - // account for trailing newlines, on all but the very last line - if ( lineLength != 0 ) - { - int lines = ( outputLength + lineLength -1 ) / lineLength - 1; - if ( lines > 0 ) - { - outputLength += lines * lineSeparator.length(); - } - } - - // must be local for recursion to work. - StringBuffer sb = new StringBuffer( outputLength ); - - // must be local for recursion to work. - int linePos = 0; - - // first deal with even multiples of 3 bytes. - int len = (b.length / 3) * 3; - int leftover = b.length - len; - for ( int i=0; i lineLength ) - { - if ( lineLength != 0 ) - { - sb.append(lineSeparator); - } - linePos = 4; - } - - // get next three bytes in unsigned form lined up, - // in big-endian order - int combined = b[i+0] & 0xff; - combined <<= 8; - combined |= b[i+1] & 0xff; - combined <<= 8; - combined |= b[i+2] & 0xff; - - // break those 24 bits into a 4 groups of 6 bits, - // working LSB to MSB. - int c3 = combined & 0x3f; - combined >>>= 6; - int c2 = combined & 0x3f; - combined >>>= 6; - int c1 = combined & 0x3f; - combined >>>= 6; - int c0 = combined & 0x3f; - - // Translate into the equivalent alpha character - // emitting them in big-endian order. - sb.append( valueToChar[c0]); - sb.append( valueToChar[c1]); - sb.append( valueToChar[c2]); - sb.append( valueToChar[c3]); - } - - // deal with leftover bytes - switch ( leftover ) - { - case 0: - default: - // nothing to do - break; - - case 1: - // One leftover byte generates xx== - // Start a new line if next 4 chars won't fit on the current line - linePos += 4; - if ( linePos > lineLength ) - { - - if ( lineLength != 0 ) - { - sb.append(lineSeparator); - } - linePos = 4; - } - - // Handle this recursively with a faked complete triple. - // Throw away last two chars and replace with == - sb.append(encode(new byte[] {b[len], 0, 0} - ).substring(0,2)); - sb.append("=="); - break; - - case 2: - // Two leftover bytes generates xxx= - // Start a new line if next 4 chars won't fit on the current line - linePos += 4; - if ( linePos > lineLength ) - { - if ( lineLength != 0 ) - { - sb.append(lineSeparator); - } - linePos = 4; - } - // Handle this recursively with a faked complete triple. - // Throw away last char and replace with = - sb.append(encode(new byte[] {b[len], b[len+1], 0} - ).substring(0,3)); - sb.append("="); - break; - - } // end switch; - - if ( outputLength != sb.length() ) - { - System.out.println("oops: minor program flaw: output length mis-estimated"); - System.out.println("estimate:" + outputLength); - System.out.println("actual:" + sb.length()); - } - return sb.toString(); - }// end encode - - /** - * decode a well-formed complete Base64 string back into an array of bytes. - * It must have an even multiple of 4 data characters (not counting \n), - * padded out with = as needed. - */ - public byte[] decodeAuto( String s) { - int nRemain = s.length() % 4; - if (nRemain == 0) { - return this.decode(s); - } else { - return this.decode(s + new String(this.charToPad, 0, 4 - nRemain)); - } - } - - /** - * decode a well-formed complete Base64 string back into an array of bytes. - * It must have an even multiple of 4 data characters (not counting \n), - * padded out with = as needed. - */ - public byte[] decode( String s) - { - - // estimate worst case size of output array, no embedded newlines. - byte[] b = new byte[(s.length() / 4) * 3]; - - // tracks where we are in a cycle of 4 input chars. - int cycle = 0; - - // where we combine 4 groups of 6 bits and take apart as 3 groups of 8. - int combined = 0; - - // how many bytes we have prepared. - int j = 0; - // will be an even multiple of 4 chars, plus some embedded \n - int len = s.length(); - int dummies = 0; - for ( int i=0; i>>= 8; - b[j+1] = (byte)combined; - combined >>>= 8; - b[j] = (byte)combined; - j += 3; - cycle = 0; - break; - } - break; - } - } // end for - if ( cycle != 0 ) - { - throw new ArrayIndexOutOfBoundsException ("Input to decode not an even multiple of 4 characters; pad with =."); - } - j -= dummies; - if ( b.length != j ) - { - byte[] b2 = new byte[j]; - System.arraycopy(b, 0, b2, 0, j); - b = b2; - } - return b; - - }// end decode - - /** - * determines how long the lines are that are generated by encode. - * Ignored by decode. - * @param length 0 means no newlines inserted. Must be a multiple of 4. - */ - public void setLineLength(int length) - { - this.lineLength = (length/4) * 4; - } - - /** - * How lines are separated. - * Ignored by decode. - * @param lineSeparator may be "" but not null. - * Usually contains only a combination of chars \n and \r. - * Could be any chars not in set A-Z a-z 0-9 + /. - */ - public void setLineSeparator(String lineSeparator) - { - this.lineSeparator = lineSeparator; - } - - /** - * Marker value for chars we just ignore, e.g. \n \r high ascii - */ - static final int IGNORE = -1; - - /** - * Marker for = trailing pad - */ - static final int PAD = -2; - - /** - * used to disable test driver - */ - private static final boolean debug = true; - - /** - * debug display array - */ - public static void show (byte[] b) - { - int count = 0; - int rows = 0; - - - for ( int i=0; i 'a'..'z' + for ( int i='a'; i<='z'; i++ ) { + this.valueToChar[index++] = (char)i; + } + + // 52..61 -> '0'..'9' + for ( int i='0'; i<='9'; i++) { + this.valueToChar[index++] = (char)i; + } + + this.valueToChar[index++] = chPlus; + this.valueToChar[index++] = chSplash; + + // build translate defaultCharToValue table only once. + for ( int i=0; i<256; i++ ) + { + this.charToValue[i] = IGNORE; // default is to ignore + } + + for ( int i=0; i<64; i++ ) + { + this.charToValue[this.valueToChar[i]] = i; + } + + this.charToValue[chPad] = PAD; + java.util.Arrays.fill(this.charToPad, chPad); + } + + /** + * Encode an arbitrary array of bytes as Base64 printable ASCII. + * It will be broken into lines of 72 chars each. The last line is not + * terminated with a line separator. + * The output will always have an even multiple of data characters, + * exclusive of \n. It is padded out with =. + */ + public String encode(byte[] b) throws IOException + { + // Each group or partial group of 3 bytes becomes four chars + // covered quotient + int outputLength = ((b.length + 2) / 3) * 4; + + // account for trailing newlines, on all but the very last line + if ( lineLength != 0 ) + { + int lines = ( outputLength + lineLength -1 ) / lineLength - 1; + if ( lines > 0 ) + { + outputLength += lines * lineSeparator.length(); + } + } + + // must be local for recursion to work. + StringBuffer sb = new StringBuffer( outputLength ); + + // must be local for recursion to work. + int linePos = 0; + + // first deal with even multiples of 3 bytes. + int len = (b.length / 3) * 3; + int leftover = b.length - len; + for ( int i=0; i lineLength ) + { + if ( lineLength != 0 ) + { + sb.append(lineSeparator); + } + linePos = 4; + } + + // get next three bytes in unsigned form lined up, + // in big-endian order + int combined = b[i+0] & 0xff; + combined <<= 8; + combined |= b[i+1] & 0xff; + combined <<= 8; + combined |= b[i+2] & 0xff; + + // break those 24 bits into a 4 groups of 6 bits, + // working LSB to MSB. + int c3 = combined & 0x3f; + combined >>>= 6; + int c2 = combined & 0x3f; + combined >>>= 6; + int c1 = combined & 0x3f; + combined >>>= 6; + int c0 = combined & 0x3f; + + // Translate into the equivalent alpha character + // emitting them in big-endian order. + sb.append( valueToChar[c0]); + sb.append( valueToChar[c1]); + sb.append( valueToChar[c2]); + sb.append( valueToChar[c3]); + } + + // deal with leftover bytes + switch ( leftover ) + { + case 0: + default: + // nothing to do + break; + + case 1: + // One leftover byte generates xx== + // Start a new line if next 4 chars won't fit on the current line + linePos += 4; + if ( linePos > lineLength ) + { + + if ( lineLength != 0 ) + { + sb.append(lineSeparator); + } + linePos = 4; + } + + // Handle this recursively with a faked complete triple. + // Throw away last two chars and replace with == + sb.append(encode(new byte[] {b[len], 0, 0} + ).substring(0,2)); + sb.append("=="); + break; + + case 2: + // Two leftover bytes generates xxx= + // Start a new line if next 4 chars won't fit on the current line + linePos += 4; + if ( linePos > lineLength ) + { + if ( lineLength != 0 ) + { + sb.append(lineSeparator); + } + linePos = 4; + } + // Handle this recursively with a faked complete triple. + // Throw away last char and replace with = + sb.append(encode(new byte[] {b[len], b[len+1], 0} + ).substring(0,3)); + sb.append("="); + break; + + } // end switch; + + if ( outputLength != sb.length() ) + { + System.out.println("oops: minor program flaw: output length mis-estimated"); + System.out.println("estimate:" + outputLength); + System.out.println("actual:" + sb.length()); + } + return sb.toString(); + }// end encode + + /** + * decode a well-formed complete Base64 string back into an array of bytes. + * It must have an even multiple of 4 data characters (not counting \n), + * padded out with = as needed. + */ + public byte[] decodeAuto( String s) { + int nRemain = s.length() % 4; + if (nRemain == 0) { + return this.decode(s); + } else { + return this.decode(s + new String(this.charToPad, 0, 4 - nRemain)); + } + } + + /** + * decode a well-formed complete Base64 string back into an array of bytes. + * It must have an even multiple of 4 data characters (not counting \n), + * padded out with = as needed. + */ + public byte[] decode( String s) + { + + // estimate worst case size of output array, no embedded newlines. + byte[] b = new byte[(s.length() / 4) * 3]; + + // tracks where we are in a cycle of 4 input chars. + int cycle = 0; + + // where we combine 4 groups of 6 bits and take apart as 3 groups of 8. + int combined = 0; + + // how many bytes we have prepared. + int j = 0; + // will be an even multiple of 4 chars, plus some embedded \n + int len = s.length(); + int dummies = 0; + for ( int i=0; i>>= 8; + b[j+1] = (byte)combined; + combined >>>= 8; + b[j] = (byte)combined; + j += 3; + cycle = 0; + break; + } + break; + } + } // end for + if ( cycle != 0 ) + { + throw new ArrayIndexOutOfBoundsException ("Input to decode not an even multiple of 4 characters; pad with =."); + } + j -= dummies; + if ( b.length != j ) + { + byte[] b2 = new byte[j]; + System.arraycopy(b, 0, b2, 0, j); + b = b2; + } + return b; + + }// end decode + + /** + * determines how long the lines are that are generated by encode. + * Ignored by decode. + * @param length 0 means no newlines inserted. Must be a multiple of 4. + */ + public void setLineLength(int length) + { + this.lineLength = (length/4) * 4; + } + + /** + * How lines are separated. + * Ignored by decode. + * @param lineSeparator may be "" but not null. + * Usually contains only a combination of chars \n and \r. + * Could be any chars not in set A-Z a-z 0-9 + /. + */ + public void setLineSeparator(String lineSeparator) + { + this.lineSeparator = lineSeparator; + } + + /** + * Marker value for chars we just ignore, e.g. \n \r high ascii + */ + static final int IGNORE = -1; + + /** + * Marker for = trailing pad + */ + static final int PAD = -2; + + /** + * used to disable test driver + */ + private static final boolean debug = true; + + /** + * debug display array + */ + public static void show (byte[] b) + { + int count = 0; + int rows = 0; + + + for ( int i=0; i