2007年12月30日日曜日

[Java>Original]Useful Network Utility(Utility to check Netmask, Network address)

Useful Network Utility to calculate CIDR address and check network address.

References
  • about CIDR
    http://h50146.www5.hp.com/products/software/oe/tru64unix/manual/v51a_ref/HTML/MAN/MAN7/0127____.HTM
  • Sample Source code
    http://osdir.com/ml/jakarta.james.devel/2003-02/msg00469.html
    http://lenya.apache.org/apidocs/1.2/org/apache/lenya/net/InetAddressUtil.html
Original Source Code
  • http://groups.google.com/group/taapps-sourcecode-libraries/web/Net.java
Check Utility to contain the ip address into the specified CIDR network address.
  • Description
    Network address check utility
    Check whether the specified ip address is into the CIDR network address(format like "192.168.1.0/24") or not.
    This method has 2 arguments, one is String object contains CIDR network address, other is IP address you want to check.
  • Sample
    //for example, my ip address is 192.168.1.3
    //check whether my ip address "192.168.1.3" is into the network 192.168.1.0/24 or not
    boolean compare=Net.containIp("192.168.1.0/24","192.168.1.3");
    if(compare){
    //contain the specified ip address "192.168.1.3" into network "192.168.1.0/24"
    }
Check Utility 2 to contain the ip address into the specified CIDR network list.
  • Description
    Check whether the specified ip address is into the CIDR network address list or not.
    This method has 2 arguments, one is List object contains CIDR network addresses, other is IP address you want to check.
  • Sample
    List networkList=new ArrayList();
    networkList.add("192.168.1.0/24");
    networkList.add("192.168.2.0/24");
    networkList.add("192.168.3.0/24");
    networkList.add("192.168.4.0/24");
    boolean compare2=Net.containIp(networkList,"192.168.2.200");
    if(compare){
    //contain the specified ip address "192.168.1.3" into network "192.168.1.0/24"
    }

2007年12月29日土曜日

[Java>Original]Useful Date Utility(Utility to Convert Date to String, Convert String to Date)

I created useful Useful Date Utility including some functions, utility to convert Date to String and convert String to Date easily.

Original Source Code
  • http://taapps-sourcecode-libraries.googlegroups.com/web/Dat.java
Converter from Date object to String easily
  • Description
    This function is a converter from Date object to String.This function works like ORACLE PL/SQL's to_char function.
    Dat.convertDateToStr method has 2 arguments, one is a Date format you want to convert, other is a Data format String object like "yyyy/MM/dd".
  • Sample
    Output Date object to String changed format like "2007/12/31 12:00:00".
    This function works like ORACLE PL/SQL's to_date function.
       java.util.Date testDate=new java.util.Date();
    .......
    System.out.println("Date Output:"+Dat.convertDateToStr(testDate,"yyyy/MM/dd HH:mm:ss"));
Converter from String object to Date easily
  • Description
    This function is a converter from String object to Date object.This function works like ORACLE PL/SQL's to_date function.
    Dat.convertStrToDate method has 2 arguments, one is a String object including date, other is a Date format String.
  • Sample
       String testDateStr="2007/12/31";
    ....
    java.util.Date convertedDate=Dat.convertStrToDate(testDateStr,"yyyy/MM/dd");
Get the long value between 2 Date objects.
  • Description
    You sometimes need to get long value between 2 Date objects(for example, you need to get execution time when debugging the program and checking performance).
    Dat.getDateDiff method has 3 arguments and returns the difference long value between 2 Date object. one argument is a first Date object, second argument is a Date object, third argument is a boolean type argument to convert the result long value to absolute value.
  • Sample
       java.util.Date date1=Dat.convertStrToDate("2007/12/31","yyyy/MM/dd");
    ..........
    java.util.Date date2=Dat.convertStrToDate("1973/12/31","yyyy/MM/dd");
    ..........
    long dateDiff=Dat.getDateDiff(date1,date2,false);

[Java>Original]Useful Object Utility(resource close function to avoid resource leak, and null and empty checker)

I created useful Object Utility including some functions, resource close function to avoid resource leak, and null and empty checker.

Original Source Code.
  • http://taapps-sourcecode-libraries.googlegroups.com/web/Obj.java
Object null and empty checker
  • Description
    We sometimes need to check whether the java object is null before accessing the object to avoid occuring the NullPointerException, and then we check whether the object is empty if the object is not null as follows.
    String test="test";
    ....
    if(test!=null && !test.equals("")){
    ....
    }
    This is a long and mistakable, and Obj.isNull method returns the value of true if the first argument object is null (not initialized),Obj.isNotNull(obj,true) method returns the value of !Obj.isNull(obj,true) like this.
    String test="test";
    ....
    if(Obj.isNull(test,true)){
    ....
    }
    The second argument is used for checking the object value. The second argument is set to true, System checks whether the first argument object is empty or not if the first argument object is defined as String. If the first argument is defined as Object[], Map, or Collection, System checks the Object has any elements more than 1. If Object doesn't have any elements, Obj.isNull(obj,true) returns true.

    • Normal Object
      Obj.isNull(normalObject,false) means if(normalObject==null) return true
    • Check String object
      Obj.isNull(stringObject, true) means if(stringObject==null || stringObject.equals("")) return true.
    • Check Array
      Obj.isNull(objects[], true) means if(objects==null || objects.length<1)>
    • Check Map Object
      Obj.isNull(mapObject,true) means if(mapObject==null || mapObject.size()<1)>
    • Check Collection Object
      Obj.isNull(listObject,true) means if(listObject==null || listObject.size()<1)>
Resource close function to avoid resource leak
  • Description
    For example, we have to close the opened stream or connection after we finished to use java.io.FileInputStream or java.sql.Connection like this.
    FileInputStream   fis=null;
    InputStreamReader isr=null;
    BufferedReader br =null;
    try{
    fis=new FileInputStream("/tmp/test.txt");
    ....
    }catch(Exception e){
    ......
    }finally{
    if(br!=null){
    try{
    br.close();
    }catch(Exception ignore){}
    }
    if(isr!=null){
    try{
    isr.close();
    }catch(Exception ignore){}
    }
    if(fis!=null){
    try{
    br.close();
    }catch(Exception e){}
    }
    }
    Before closing the FileInputStream, InputStreamReader, BufferedReader in finally block, we usually check whether the object is null or not, and then we close the opened object in try-catch exception handler to catch the Exception when closing the opened object.
    These statements are troublesome and mistakable. We can change the statements described above into following. We don't need to check whether the object is null or not and catch the Exception occuring while closing the object.
    FileInputStream   fis=null;
    InputStreamReader isr=null;
    BufferedReader br =null;
    try{
    fis=new FileInputStream("/tmp/test.txt");
    ....
    }catch(Exception e){
    ......
    }finally{
    Obj.close(fis);
    Obj.close(isr);
    Obj.close(br);
    }
  • We can use this Obj.close function for these class.
    1. java.io.Reader object
    2. java.io.InputStream
    3. java.io.OutputStream
    4. javax.imageio.ImageOutputStream
    5. javax.imageio.ImageInputStream
    6. javax.imageio.ImageWriter
    7. java.awt.Graphics
    8. java.io.Writer
    9. java.sql.Connection
    10. java.sql.Statement
    11. java.sql.ResultSet
    12. java.net.HttpURLConnection
    13. java.net.URLConnection

2007年12月28日金曜日

[Java>Norbert]Norbert Utility(robots.txt parser) sometimes fails to check robots.txt properly

References
  1. Norbert Page(parser for robots.txt file)
    http://osjava.org/norbert/
  2. About specification of robots.txt file
    http://www.robotstxt.org/wc/norobots-rfc.html
  3. About robots.txt by Baidu(japanese)
    http://www.baidu.jp/search/robots.html
  4. list of Crawlers
    http://www.robotstxt.org/wc/active/html/index.html
Description
Norbert is a parser and utility to find and check the robots.txt file in web page.
(If you need to get the norbert utility, please access to http://osjava.org/norbert/ described above.)
But Norbert utility sometimes doesn't check and parse the robots.txt file properly even though the robots.txt file is existed in top directory of the web site. Norbert checks the string case sensitive, but some web sites ignore the tag's case sensitive in robots.txt file(refer to http://www.robotstxt.org/wc/norobots-rfc.html) and Norbert utility failed to check the line and can't pick up the tags properly.

Solution
You can avoid this problem if you change parseTextForUserAgent function in org.osjava.norbert.NoRobotClient.java as follows.
Before checking the tags in robots.txt, System changes all of the line's text to lowercase, and then System uses the String object converted to lowercase.

Customization Example
   
private RulesEngine parseTextForUserAgent(String txt, String userAgent) throws NoRobotException {
.....
try {
while( (line = rdr.readLine()) != null ) {
// trim whitespace from either side
line = line.trim();

//change the line's string to new lowercase string
//add this line to store the text after exchanging all of the line's text to lowercase.
String lineToLowerCase=line.toLowerCase();
//end
.....
//replace the "line" object with lineToLowerCase string object when checking the line string.
if(lineToLowerCase.startsWith("user-agent:")) {

}
else{
//replace using the line object with lineToLowerCase string object.
if(lineToLowerCase.startsWith("allow:")) {
.....
} else
//replace using the line object with lineToLowerCase string object.
if(lineToLowerCase.startsWith("disallow:")) {
.....
} else {
.....
}
}
.....
}

2007年12月25日火曜日

[Java>Basic Authorization]Sample implementation of Basic Authorization Logic in JavaServlet manually.

References
  1. About WWW-Authenticate, Authorization header
    http://www.tohoho-web.com/ex/http.htm#WWW-Authenticate
    http://www.tohoho-web.com/ex/http.htm#Authorization
  2. Basic Authentificator Sample written in Java
    http://www.katch.ne.jp/~h-inoue/tips/javaservlet/0001.html
  3. About Digest Authorization(Japanese)
    https://www.codeblog.org/blog/inoue/20060227.html
    http://shain.tomocreative.net/2007/03/phpdigest.html(for PHP)
  4. MD5, Base64 useful Library
    ostermiller utility, http://ostermiller.org/
Customization Example

/*
Created by Tatsuya Anno
Sample implementation of Basic Authorization Logic in JavaServlet manually.
*/
String authorizationMethod="Basic";
String wwwAuthenticateValue="BASIC realm=\"Basic Auth Test\"";

//check authorization header
//Get Authorization Header information
//Authorization header is encoded in Http Header based on following format.
//"BASIC [Username]:[Password]", value of [Username]:[Password] is encoded by Base64.
//Sample value of authorization header is "Basic YW50YXRzdTp0ZXN".
String authorization=request.getHeader("Authorization");
if(authorization==null||authorization.equals("")){
//if there is no Authorization Header, return SC_UNAUTHORIZED status.
response.setHeader("WWW-Authenticate",wwwAuthenticateValue);
response.sendError(HttpServletResponse.SC_UNAUTHORIZED);
return null;
}
else{
if(!authorization.startsWith(authorizationMethod)){
//If there is unexpected authorization header, return SC_UNAUTHORIZED status.
//If we use basic auth logic, the vlaue of authorization header starts with "Basic".
response.setHeader("WWW-Authenticate",wwwAuthenticateValue);
return null;
}

//Remove "BASIC" characters from the value of Authorization Header
//before decoding base64 value to Username:Password value.
String encodedHeader=authorization.substring(6);

//decode BASE64 characters
//base64 decoder by using sun.misc.BASE64Decoder
//but sun.misc.BASE64Decoder class is not public.
//(Please refer to http://www.source-code.biz/snippets/java/2.htm)
//sun.misc.BASE64Decoder decoder = new sun.misc.BASE64Decoder();
//byte[] decodedBytes = decoder.decodeBuffer(encodedHeader);
//----
//I changed the base64 utility from sun.misc.BASE64Decoder to ostermiller utility.
//decode base64 encoded characters by using Ostermiller Utility.
//Please refer to the http://ostermiller.org/
byte[] decodedBytes=com.Ostermiller.util.Base64.decodeToBytes(encodedHeader);
if(decodedBytes==null){
//If the decoded byte values is null, return UNAUTHORIZED status
response.setHeader("WWW-Authenticate",wwwAuthenticateValue);
response.sendError(HttpServletResponse.SC_UNAUTHORIZED);
return null;
}

//create new string based on the byte values decoded from base64
String decodedAuthHeader=new String(decodedBytes);
if(decodedAuthHeader==null||decodedAuthHeader.equals("")){
response.setHeader("WWW-Authenticate",wwwAuthenticateValue);
response.sendError(HttpServletResponse.SC_UNAUTHORIZED);
return null;
}
//get username and password from decoded string object.
//split decoded string which is separated by ":".
String[] authTokens=decodedAuthHeader.split(":");
if(authTokens.length<2){
response.setHeader("WWW-Authenticate",wwwAuthenticateValue);
response.sendError(HttpServletResponse.SC_UNAUTHORIZED);
return null;
}

//If system fails to Split string, return SC_UNAUTHORIZED status because of unexpected.
if(authTokens[0]==null||authTokens[0].equals("")){
response.setHeader("WWW-Authenticate",wwwAuthenticateValue);
response.sendError(HttpServletResponse.SC_UNAUTHORIZED);
return null;
}

//check the values of username and password, following is a sample logic.
if(!authTokens[0].equalsIgnoreCase("auth_test")||!authTokens[1].equals("12345"))
{
//if fails to check username and password, return SC_UNAUTHORIZED status
response.setHeader("WWW-Authenticate",wwwAuthenticateValue);
response.sendError(HttpServletResponse.SC_UNAUTHORIZED);
return null;
}
}
//end of manual basic authorization

2007年12月22日土曜日

[Java>JavaMail] Sometimes "java.net.ConnectException: Connection refused" occures.

[Description]
I'm using Sendmail SMTP Server in our Web System.
Sometimes in our web systems, "java.net.ConnectException: Connection refused" were occured when accessing SMTP Server via Java Servlet Programs. I tried to reproducing this problem again and again, but I could not reproduced this problem manually. I tried to simulate the operations to reproduce this problem in parallel by using Apache JMeter, I could reproduce this problem .

[Error Details]
java.net.ConnectException: Connection refused -> final: xxxxx.jp(Domainname): javax.mail.MessagingException: Could not connect to SMTP host: localhost, port: ;

And I got the result of executing ps command as follows to check the sendmail process when this problem occured.
xxxxx 21249 1 0 12:22 ?
00:00:00 sendmail: rejecting connections on daemon MTA: load average: 13

[Cause]
In case of this, this problem occured only when the CPU load average was high.
We are using Sendmail and qpopper as SMTP system, Sendmail has a functionality to reject connecting to the SMTP system.

[Workaround]
We can control rejecting the user's connection by using following parameters when CPU load is high. We set these parameters into sendmail.mc, try recreating sendmail.cf and restart sendmail process.
define(`confQUEUE_LA',`30')dnl
define(`confDELAY_LA',`40')dnl
define(`confREFUSE_LA',`100')dnl

2007年12月17日月曜日

[JAVA>Velocity, Template]customization how to convert String to Template Object

References
  1. Apache Velocity Home: http://velocity.apache.org/
Description
I added following customization logic which convert String java object to Velocity Template object because I sometimes needed template object dynamically without creating template file.
(Please let me know if you have any other good ideas.)

Customization Example Steps
  1. get Velocity Source code from velocity.apache.org site.
  2. open org.apache.velocity.Template.java file.
  3. add java class import statements in Template.java
    import org.apache.velocity.runtime.RuntimeSingleton;
    import org.apache.velocity.runtime.RuntimeConstants;
    import java.io.StringReader;
  4. add customized constructor function of Template object as follows.
    /*
    customized constructor which convert String Object to Template Object.
    @param pTemplateName is an template name
    @param pTemplateString is String object you want to render
    @return a velocity template object.
    */
    public Template(String pTemplateName,String pTemplateString)
    throws Exception
    {
    //Initialization
    RuntimeSingleton.init();

    setRuntimeServices(RuntimeSingleton.getRuntimeServices());
    setName(pTemplateName);
    //set encodeing if you need
    //setEncoding(RuntimeConstants.ENCODING_DEFAULT);

    data = null;
    errorCondition = null;
    StringReader stringReader=null;
    BufferedReader bufferedReader=null;

    try{
    stringReader=new StringReader(pTemplateString);
    bufferedReader=new BufferedReader(stringReader);

    data = rsvc.parse(bufferedReader,name);
    initDocument();
    }catch(Exception e){
    errorCondition=new Exception("Error occured while initializaing String Template, customization");
    throw errorCondition;
    }finally{
    if(stringReader!=null){
    stringReader.close();
    }
    if(bufferedReader!=null){
    bufferedReader.close();
    }
    }
    }
    /*
    End of Customization
    */
  5. compile new java file.
how to use this customization
String testTemplate="Velocity Template Test";
Template template=new Template("TemplateName",testTemplate);

2007年12月15日土曜日

[JAVA>JML, MSN Messenger] When using JML(Java MSN Messenger Library), NullPointerException occured

References
  1. Reference Site
    Java MSN Messenger Library (JML) Home: http://java-jml.sourceforge.net/
Problems
In case of using Java MSN Messenger Library(JML), NullPointerException occures when adding a friend into Friend List or removing a friend from Friend List, but this problem doesn't always occured.

Reason why this problem occures
System checks whether a friend is already added into the list or not before adding or removing a friend. When checking a friend, System accesses the user's contact list object and get contact information from the contact list object.
When a friend doesn't contain the user's contact list, system returns the null value. And then, system accesses the object even though the value of the contact object is null.

Solution Example
Change addFriend() and removeFriend() function int the net.sf.jml.impl.SimpleMessenger.java file as follows. You need to check whether the value of MsnContact object is nujll or not.
  1. addFriend function

    private void addFriend(MsnList list, Email email, String friendlyName) {
    if (list == null || email == null || list == MsnList.RL
    || list == MsnList.PL)
    return;
    MsnContact contact = getContactList().getContactByEmail(email);
    /*
    * added by tatsuya anno to avoid NullPointerException
    * because contact object is null when adding new user
    */
    if(contact!=null){
    if (contact.isInList(list)){
    return;
    }
    }
    /*
    * End of Bugfix
    */
    .......
    }


  2. removeFriend function

    private void removeFriend(MsnList list, Email email, String id,
    String groupId) {
    if (list == null || list == MsnList.RL)
    return;
    if (list == MsnList.FL) {
    if (id == null)
    return;
    } else if (email == null)
    return;
    MsnContact contact = getContactList().getContactByEmail(email);

    /*
    * added by tatsuya anno to avoid NullPointerException
    * because contact object is null when adding new user
    */
    if(contact!=null){
    if (!contact.isInList(list)){
    return;
    }
    }
    /*
    * End of Bugfix
    */
    ......
    }

2007年6月19日火曜日

How to translate Openfire Xmpp Server in Japanese

How to translate Openfire Xmpp Server in Japanese

Reference, Translation Guide
  • http://www.igniterealtime.org/builds/openfire/docs/latest/documentation/translator-guide.html

Download Japanese Message Properties File
I put 2 Japanese translated properties file.(These files are based on Openfire Version3.3.1 properties)
One is converted to ASCII format(\uXXXX format by using native2ascii command), other is saved messages based on UTF-8 characterset without ASCII format conversion.we usually use only converted japanese messages in ASCII format. Please downlod following file.
  1. converted japanese messages in ASCII format
    http://www.box.net/shared/ou0niqzp19
  2. UTF-8 no bom file without converting
    http://www.box.net/shared/z1euxze9v3
Steps to install file
If you need to translate all messages including sidebar and tag, please try following steps.
(Please refer to following URL if you need more information
http://www.igniterealtime.org/forum/thread.jspa?threadID=27081&tstart=0)
  1. Download converted japanese messages in ASCII formt file descrived above.
  2. Change the file name from openfire_i18n_ja_JP.properties.converted to openfire_i18n_ja_JP.properties.
  3. Create a new jar file.
    jar cvf cfja.jar openfire_i18n_ja_JP.properties
  4. Put new cfja.jar file into the lib directory under your Openfire Home directory.
  5. Restart your openfire server.
Following steps are easy to use, but some messages(in sidebar and tag) are displayed in English.
  1. Download Ascii converted japanese message file.
  2. Put the file into $OpenfireHome/lib named as openfire_i18n_ja_JP.properties.
  3. Open conf/openfire.xml and change locate tag manually to ja_JP
  4. Restart openfire server.
You can translate openfire by using following steps, but it is not easy.
(Changed at 2007/06/20)
  1. Backup lib/openfire.jar
  2. Extract openfile.jar
  3. Copy Ascii converted japanese message file into the extracted directory.
  4. Create new jar file.
  5. Put created jar file into lib directory.
  6. Restart openfire server.

2007年6月8日金曜日

[JAVA>Rome, RSS/XML] When using Rome RSS Reader or Rome Fetcher(XML Reader), MalformedInputException occured

References
  1. Rome WebSite
    https://rome.dev.java.net/
Problems
When using Rome RSS Reader or Rome Fetcher to get RSS Data from any Website, sometimes sun.io.MalformedInputException error occured randomly. In case of fetching RSS from same Website, sometimes error occured, but sometimes not occured. Error stacks were as follows.

sun.io.MalformedInputException
at sun.io.ByteToCharUTF8.flush(ByteToCharUTF8.java:139)
at sun.nio.cs.StreamDecoder$ConverterSD.flushInto(StreamDecoder.java:333)
at sun.nio.cs.StreamDecoder$ConverterSD.implRead(StreamDecoder.java:357)
at sun.nio.cs.StreamDecoder.read(StreamDecoder.java:250)
at java.io.InputStreamReader.read(InputStreamReader.java:212)
at java.io.BufferedReader.fill(BufferedReader.java:157)
at java.io.BufferedReader.readLine(BufferedReader.java:320)
at java.io.BufferedReader.readLine(BufferedReader.java:383)
at com.sun.syndication.io.XmlReader.getXmlProlog(XmlReader.java:534)
at com.sun.syndication.io.XmlReader.doHttpStream(XmlReader.java:325)
at com.sun.syndication.io.XmlReader.(XmlReader.java:248)


Reason why this errr occured
While finding the Character Encoding in RSS InputStream in getXmlProlog() function In XmlReader.java(rome-0.9/src/java/com/sun/syndication/io/XmlReader.java), System tries to get first 4096 bytes characters from Stream. In case of including Japanese or not Ascii characters, System sometimes truncates the bytes at the invalid byte position(Japanese uses 3 byte for each word in UTF-8), so this problem occures.

Solution
  • Fixed file
    rome-0.9/src/java/com/sun/syndication/io/XmlReader.java
  • Fixed function
    Change as follows in getXmlProlog() function.
  • Description
    We catches the MalformedInputException exception and ignore occuring this exception because the line string this exception occures doesn't usually need. Add try and catch block when readling the line.
  • Example

    /** Changed by tatsuya anno to avoid MalformedInputException*/
    try{
    String line = br.readLine();
    while (line != null) {
    prolog.append(line).append("\n");
    line = br.readLine();
    }
    }catch(sun.io.MalformedInputException eMalformedInput){
    //skip
    }


2007年6月5日火曜日

[JAVA>JYMSG]How to use JYMSG Java Library for Yahoo Japan

How to use JYMSG Java Library for Yahoo Japan

References
  • JYMSG Homepage: http://jymsg9.sourceforge.net/

Description
Now, the messenger server for Yahoo Japan is separated from Yahoo US, and the some spec of messenger client for Yahoo Japan is different from the client for Yahoo US. While implementing messenger for Yahoo Japan,I had a character encoding problem.

Only when sending a message from Yahoo Japan client to Yahoo Japan Messenger Server, Yahoo Japan Messenger Client is sending a packet encoded by using SJIS characterset (not using UTF-8).
This problem is occured only when using Yahoo Japan Client, not Yahoo US Client. (I don't know why yahoo japan uses SJIS character encoding.) I customized the JYMSG library source code for supporting Yahoo Japan as follows.

customization steps
  1. added following in YMSG9InputStream.java to handle user defined encoding.

    private String characterEncoding="UTF-8";
    public void setCharacterEncoding(String pCharacterEncoding){
    if(pCharacterEncoding==null||pCharacterEncoding.equals("")){
    return;
    }
    characterEncoding=pCharacterEncoding;
    }
    public String getCharacterEncoding(){
    return characterEncoding;
    }

  2. at 68 in YMSG9InputStream.java, change hardcoded UTF-8 characterset to variable named characterEncoding

    for(int i=0; i<body.length-1;i++)
    { if(u2i(body[i])==0xc0 && u2i(body[i+1])==0x80)
    { // -----Create a UTF-8 string and add to array
    //changed by tatsuya anno
    Commented out ==> //String s = new String(body,start,i-start,"UTF-8");
    added new line ==> String s=new String(body,start,i-start,characterEncoding);
    //end of customization
    v.addElement(s);
    if(dbis!=null) System.out.println(">>"+s);
    // -----Skip over second byte in separator, reset string start
    i++; start=i+1;
    }
    }

  3. added following in DirectConnectionHandler.java

    //added by tatsuya anno
    public void setInputStreamCharacterEncoding(String pCharacterEncoding){
    if(ips==null){
    return;
    }

    ips.setCharacterEncoding(pCharacterEncoding);
    }

  4. example using this customization

    //create ConnectionHander object to connect to yahoo japan
    DirectConnectionHandler connectionHandler=new DirectConnectionHandler("",);
    //create Session object
    Session yahooConnection=new Session(connectionHandler);
    yahooConnection.addSessionListener(new YahooPacketListener());
    yahooConnection.login(yahooClientUsername,yahooClientPassword);
    yahooConnection.setStatus(StatusConstants.STATUS_AVAILABLE);

    //after logged in, Set InputStream Encoding Charset
    //if you use this library for Yahoo US, you don't need to this line.
    connectionHandler.setInputStreamCharacterEncoding("SJIS");