FileDocCategorySizeDatePackage
MMSAddress.javaAPI DocphoneME MR2 API (J2ME)31180Wed May 02 18:00:44 BST 2007com.sun.tck.wma.mms

MMSAddress

public class MMSAddress extends Object
MMS address parsing and validation. MMS address is validated according to Table D-2 in WMA2.0 specification. In addition address is parsed into the fields of MMSAddress object.

Fields Summary
public static final int
INVALID_ADDRESS
Type corresponding to invalid address.
public static final int
EMAIL
Type corresponding to the e-mail address.
public static final int
GLOBAL_PHONE_NUMBER
Type corresponding to global phone number address.
public static final int
IPV4
Type corresponding to the ipv4 address.
public static final int
IPV6
Type corresponding to the ipv6 address.
public static final int
SHORTCODE
Type corresponding to the shortcode address.
public static final int
APP_ID
Type corresponding to the application id address.
public String
address
Field that holds address part of the mms address (without "mms://" and ":" separator before app id).
public String
appId
Field that holds application id part of the mms address which appears after "mms://:" or after phone number, ipv4, or ipv6 followed by ":" .
public int
type
Type of this MMSAddress instance, INVALID_ADDRESS when uninitialized .
Constructors Summary
MMSAddress()
MMSAddress constructor to create uninitialized instance.


                
     
	clear();
    
MMSAddress(String address, String appId, int type)
MMSAddress constructor to create initialized instance.

param
address The address part of the mms address
param
appId The application id part of the mms address
param
type The type of this mms addreess (EMAIL, GLOBAL_PHONE_NUMBER, IPV4, IPV6, SHORTCODE, APP_ID)

	set(address, appId, type);
    
Methods Summary
private static booleanalpha(char c)
Determines whether a character is a letter.

param
c The character to check.
return
true if c is a letter; false otherwise

	return (c >= 'a" && c <= 'z") || (c >= 'A" && c <= 'Z");
    
private static booleanasciiControl(char c)
Determines whether a character represents an ascii control character.

param
c The character to check.
return
true if c is a character. false otherwise

	return ((int)c >= 0 && (int)c <= 31);
    
private static booleanatomChar(char c)
Determines whether a character represents an atom character. as specified in WMA 2.0 specification.

param
c The character to check.
return
true if c is a special character. false otherwise

        return ((int)c <= 127 && (int)c > 32 && !specials(c));
    
voidclear()
Clears MMSAddress fields. Type is set to INVALID_ADDRESS while address and appId are set to null.

	address = appId = null;
	type = INVALID_ADDRESS;
    
private static booleandigit(char c)
Determines whether a character represents a digit.

param
c The character to check.
return
true if c is a digit; false otherwise

	return (c >= '0" && c <= '9");
    
java.lang.StringgetMMSAddressString()
Creates a valid mms address corresponding to the values in this MMSAddress object. If the object is not intialized null is returned.

return
mms address corresponding to this MMSAddress object.


	if (type == INVALID_ADDRESS || 
	    (address == null && appId == null)) {
	    return null;
	}

	StringBuffer mmsAddr = new StringBuffer("mms://");
	if (address != null) mmsAddr.append(address);
	if (appId != null) mmsAddr.append(":").append(appId);
	return mmsAddr.toString();
    
public static com.sun.tck.wma.mms.MMSAddressgetParsedMMSAddress(java.lang.String addressStr)
Determines whether a string represents a valid mms address as specified in WMA 2.0 specification and parses the incoming string into address and application id strings.

param
addressStr The string to check.
return
newly created MMSAddress object if addressStr is a valid mms address; null otherwise

	return getParsedMMSAddress(addressStr, null);
    
public static com.sun.tck.wma.mms.MMSAddressgetParsedMMSAddress(java.lang.String addressStr, com.sun.tck.wma.mms.MMSAddress mmsAddress)
Determines whether a string represents a valid mms address as specified in WMA 2.0 specification and parses the incoming string into address and application id strings.

param
addressStr The string to check.
param
mmsAddress The return MMSAddress object which fields will be filled with parsed address and application id, and correspoinding type (EMAIL, GLOBAL_PHONE_NUMBER, IPV4, IPV6, SHORTCODE, APP_ID).
return
MMSAddress object if addressStr is a valid mms address; null otherwise


	if (addressStr == null || !addressStr.startsWith("mms://")) {
	    return null;
	}

        if (mmsAddress == null) {
            mmsAddress = new MMSAddress();
        }

	if (parsePhoneNumber(addressStr, 6, mmsAddress) || 
	    parseIpv4(addressStr, 6, mmsAddress) || 
	    parseIpv6(addressStr, 6, mmsAddress) || 
	    parseEmail(addressStr, 6, mmsAddress) ||
	    parseShortCode(addressStr, 6, mmsAddress) || 
	    parseApplicationId(addressStr, 6, mmsAddress)) {
	    return mmsAddress;
	}

	return null;
    
private static booleanhex(char c)
Determines whether a character represents a hexadecimal number. ('0' - '9' and 'A' - 'F')

param
c The character to check.
return
true if c is a hexadecimal number; false otherwise

	return (digit(c) || (c >= 'A" && c <= 'F"));
    
private static booleanisChar(char c)
Determines whether a character represents a character.

param
c The character to check.
return
true if c is a character. false otherwise

	return ((int)c) >= 0 && ((int)c) <= 127;
    
private static intisDomain(java.lang.String s, int i, int n)
Determines whether a string represents a valid "domain" starting from an index. Domain syntax is specified in WMA 2.0 spec as 1*(atom | "[" *(text w/o ",[,],\r but with linear space | "\"alpha) "]") where atom is represented by at least one char or more which is not a space or a special or control character.

param
s The string to check.
param
i The index of the string at which to start the check 0 <= i <= s.length()
param
n The index of the last character to be checked in s.
return
index till which s contains valid domain specification; -1 if index i is invalid or if characters in s starting from i do not comply with domain specification

	return isSequenceOfAtomAndText(s, i, n, '[", ']", true);
    
private static intisLocalAtDomain(java.lang.String s, int i, int n)
Determines whether a string starting from an index satisfies the following syntax: local-part"@"domain

param
s The string to check.
param
i The index of the string at which to start the check. 0 <= i <= s.length()
param
n The index of the last character to be checked in s.
return
index till which s contains valid mailbox specification; -1 if index i is invalid or if characters in s starting from i is not in the form of local-part"@"domain

        if (n > s.length()) {
            n = s.length();
        }
	if (i >= n) {
	    return -1;
	}

	// local
	if ((i = isLocalPart(s, i, n)) <= 0) {
	    return -1;
	}

	if (i == n || s.charAt(i) != '@") {
	    return -1;
	}

        i++;

	// domain
	return isDomain(s, i, n);
    
private static intisLocalPart(java.lang.String s, int i, int n)
Determines whether a string represents a valid "local-part" starting from an index. Local-part syntax is specified in WMA 2.0 spec as (atom | "\"alpha) *("." (atom | "\"alpha)) where atom is represented by at least one char which is not a space or a special or control character.

param
s The string to check.
param
i The index of the string at which to start the check 0 <= i <= s.length()
param
n The index of the last character to be checked in s.
return
index till which s contains valid local-part specification; -1 if index i is invalid or if characters in s starting from i do not comply with local-part specification

	return isSequenceOfAtomAndText(s, i, n, '"", '"", true);
    
private static intisMailbox(java.lang.String s, int i, int n)
Determines whether a string represents a valid mailbox address starting from an index. Mailbox syntax is specified in WMA 2.0 spec as local-part"@"domain or as [phrase] "<"[("@"domain) [*("," ("@"domain))]] local-part"@"domain">"

param
s The string to check.
param
i The index of the string at which to start the check. 0 <= i <= s.length()
param
n The index of the last character to be checked in s.
return
index till which s contains valid mailbox specification; -1 if index i is invalid or if characters in s starting from i do not comply with mailbox specification

        if (n > s.length()) {
            n = s.length();
        }
	if (i >= n) {
	    return -1;
	}

	int initI = i;

	if ((i = isLocalAtDomain(s, initI, n)) > 0) {
	    return i;
	}

	i = initI;

	// check for  [phrase] <[1#(@ domain):] local @ domain>
	// note phrase is optional
	if ((s.charAt(i) != '<") &&
	    (i = isPhrase(s, initI, n)) < 0) {
	    return -1;
	}

	if (i == n || s.charAt(i) != '<") {
	    return -1; 
	}

	i++;
	    
	if (i == n) {
	    return -1;
	}

	// there can be a list of domains: 
	// @ domain, @domain, @domain :
	boolean atLeastOneDomain = false;
	while (s.charAt(i) == '@") {
	    atLeastOneDomain = true;

	    if ((i = isDomain(s, i+1, n)) < 0) {
		return -1;
	    }

	    if (i == n) {
		return -1;
	    }
	    
	    if (s.charAt(i) == ':") {
		break;
	    }

	    // @domain,,@domain is allowed
	    while (s.charAt(i) == ',") {
		i++;
		if (i == n || s.charAt(i) == ':") {
		    return -1;
		}
	    }
	}

	if (atLeastOneDomain) {
	    if (i == n || s.charAt(i) != ':") {
		return -1;
	    }
	    i++;
	}

	// local @ domain
	if ((i = isLocalAtDomain(s, i, n)) <= 0) {
	    return -1;
	}

	if (i < n && s.charAt(i) == '>") {
	    return i + 1;
	}
	return -1;
    
private static intisPhrase(java.lang.String s, int i, int n)
Determines whether a string represents a valid "phrase" starting from an index. Phrase syntax is specified in WMA 2.0 spec as 1*(atom | """ *(text w/o ",\,\r but with linear space | "\"alpha) """) where atom is represented by at least one char or more which is not a space or a special or control character.

param
s The string to check.
param
i The index of the string at which to start the check 0 <= i <= s.length()
param
n The index of the last character to be checked in s.
return
index till which s contains valid phrase specification; -1 if index i is invalid or if characters in s starting from i do not comply with phrase specification

	return isSequenceOfAtomAndText(s, i, n, '"", '"", false);
    
private static intisSequenceOfAtomAndText(java.lang.String s, int i, int n, char begSep, char endSep, boolean dotSeparated)
Determines whether a string starting from an index represents a valid sequence that satisfies the following syntax: text ::== *(1*(|\ALPHA)) sequence ::== 1*(atom | text) | (atom | text) *("." (atom | text)) where atom is represented by at least one or more characters that are not space or special or control characters and begSep/endSep are separators that are placed around text. LWSP is a linear white space (space and tabs and \r\n but not \r).

param
s The string to check.
param
i The index of the string at which to start the check 0 <= i <= s.length()
param
n The index of the last character to be checked in s.
param
begSep The character marking beginning of quoted text
param
endSep The character marking the end of the text
param
dotSeparated If true atom and quoted text should be separated by a "." character; ".." is not allowed
return
index till which s contains valid sequence; -1 if index i is invalid or if characters in s starting from i do not comply with the sequence specification.

 
	if (i >= s.length()) {
	    return -1;
	}
        if (n > s.length()) {
            n = s.length();
        }

	// see if there is at least one word
	boolean withinQuotation = false;
	boolean atLeastOneWord  = false;

	char c;
	int initI = i;

	for (; i < n; i++) {
	    c = s.charAt(i);
  	    if (c == begSep) {
		if (begSep == endSep) {
		    withinQuotation = !withinQuotation;
		} else if (withinQuotation) {
		    return -1;
		} else {
		    withinQuotation = true;
		}
		atLeastOneWord = true;
	    } else if (c == endSep) {
		if (!withinQuotation) {
		    return -1;
		}
		withinQuotation = false;
	    } else if (c == '\\") {
		// chars can be quoated only within separators
		// and there has to be another char after '\'
		if (!withinQuotation || i == n-1) {
		    return -1;
		}
		i++;
		c = s.charAt(i);
		// only ALPHA character can be quoted
		if (!alpha(c)) {
		    return -1;
		}

	    } else if (dotSeparated && c == '.") {
		if (i == initI ||
		    ((i - initI) > 0 && s.charAt(i-1) == '.")) {
		    // empty word
		    // string starts with .
		    return -1;
		}
	    } else if (withinQuotation) {
		// within quotation \r is not allowed but
		// \r is allowed as part of linear-white-space
		if (c == ' " || c == '\t" || c == '\r") {
		    if ((i = linearWhiteSpaceSeq(s, i, n)) < 0) {
			return -1;
		    }
		    i--;
		}
	    } else if (atomChar(c)) {
		atLeastOneWord = true;
	    } else {
		// this is supposed to be an atom 
		break;
	    }
	}

	// open quotation or no words or
	// dot separator is the last char
	if (withinQuotation || !atLeastOneWord ||
	    (dotSeparated && i > initI && s.charAt(i-1) == '.")) {
	    return -1;
	}

	return i;
    
private static intlinearWhiteSpaceSeq(java.lang.String s, int i, int n)
Determines first not linear white space character in s after character with index i but before character with index n.

param
s The string to check.
param
i The index of the string at which to start the check. 0 <= i <= s.length()
param
n The index of the last character to be checked in s.
return
The first non linear white character after position i, it could the the same as passed in i value if there are no linear white space characters.

	char c;
	for (; i < n; i++) {
	    c = s.charAt(i);
            if (c == ' " || c == '\t") {
                continue;
            }
	    if (c == '\r") {
		// only CR LF one after another are allowed
		// CR by itself is not allowed
	        if (i == n-1 || s.charAt(i+1) != '\n") {
		    return i;
		}
		i++;
	    } else {
		return i;
	    }
	}

	return i;
    
static booleanparseApplicationId(java.lang.String s, int i, com.sun.tck.wma.mms.MMSAddress mmsAddress)
Determines whether a string represents a valid application ID starting from an index.If the address string is a valid application id representation it is parsed into the fields of MMSAddress object (application id and null for the address). Application ID syntax is specified in WMA 2.0 spec as : ":"[*(1*(alpha | digit | "." | "_") ".")]1*(alpha | digit | "." | "_"). The number of characters in application ID must not exceed 32.

param
s The string to check.
param
mmsAddress The return MMSAddress object which fields are filled with parsed address (null in this case) and application id, its type is set to APP_ID.
param
i The index of the string at which to start the check. 0 <= i <= s.length()
return
true if s represents a valid application ID starting from index i; false otherwise and if index i is invalide value

	int len = s.length();

	// empty string or string like ":" are not allowed
	// only 32 characters is allowed
        if (i <= 0 || i >= len || (len-i) > 32 || (len-i) < 1 ||
            s.charAt(i) != ':") {
	    return false;
	}
	
	i++; // to skip ':'

	char c;
	int initI = i;

	for (int j = 0; i < len; i++) {
	    
	    c = s.charAt(i);
	    
	    for (j = 0; i < len; i++, j++) {
		c = s.charAt(i);

	        if ((c != '_") && /* dot is also allowed in the spec ??? */
		    !digit(c) && !alpha(c)) {
		    break;
		}
	    }

	    if (i == len) {
		if (mmsAddress.type == INVALID_ADDRESS) {
		    mmsAddress.set(null, s.substring(initI), APP_ID);
		} else {
		    mmsAddress.setAppid(s.substring(initI));
		}
		return true;
	    }

	    // there has to be at least one digit or letter and
	    // nondigit and nonalpha char can be only dot (one dot only)
	    if (j == 0 || c != '.") {
		return false;
	    }
	}
	
	return false;
    
static booleanparseEmail(java.lang.String s, int i, com.sun.tck.wma.mms.MMSAddress mmsAddress)
Determines whether a string represents a valid e-mail address starting from an index. If the address string is a valid e-mail address representation it is parsed into the fields of MMSAddress object (address and null for appId). E-mail syntax is specified in WMA 2.0 spec.

param
s The string to check.
param
i The index of the string at which to start the check. 0 <= i <= s.length()
param
mmsAddress The return MMSAddress object which fields are filled with parsed address and application id (null in this case), its type is set to EMAIL.
return
true if s represents a valid e-mail address starting from index i; false otherwise and if index i is invalide value


	int len = s.length();
	if (i <= 0 || i >= len) {
	    return false;
	}

	int initI = i;

	// email can be on of the following:
	// emial ::== mailbox | phrase : [#mailbox] ;

	// if there is a ';' at the end then there should be a phrase
	// at the beginning
	if (s.charAt(len-1) == ';") {

	    if ((i = isPhrase(s, i, len-1)) <= 0) {
		return false;
	    }

	    // there was no ":"
	    if (i >= len-1 || s.charAt(i) != ':") {
		return false;
	    }

	    i++; // move after ':'

	    if (i == len-1) {
                mmsAddress.set(s.substring(initI), null, EMAIL);
		return true; // phrase : ; is allowed string
	    }

	    if ((i = isMailbox(s, i, len-1)) <= 0) {
		return false;
	    }

	    while (i < len-1) {

		// there has to be a ',' sep
		if (i >= len-2 && s.charAt(i) != ',") {
		    return false;
		}

		if ((i = isMailbox(s, i+1, len-1)) <= 0) {
		    return false;
		}
	    }
	    if (i == len-1) {
		mmsAddress.set(s.substring(initI), null, EMAIL);
		return true;
	    }
	    return false;
	}

	if (isMailbox(s, i, len) == len) {
	    mmsAddress.set(s.substring(initI), null, EMAIL);
	    return true;
	}
	return false;
    
static booleanparseIpv4(java.lang.String s, int i, com.sun.tck.wma.mms.MMSAddress mmsAddress)
Determines whether a string represents a valid ipv4 address starting from an index. If the address string is a valid ipv4 representation it is parsed into the fields of MMSAddress object (address and application id). ipv4 syntax is specified in WMA 2.0 spec as : 1*3digit "." 1*3digit "." 1*3digit "." 1*3digit [applicationId]

param
s The string to check.
param
mmsAddress The return MMSAddress object which fields are filled with parsed address and application id (which can be null), its type is set to IPV4.
param
i The index of the string at which to start the check. 0 <= i <= s.length()
return
true if s represents a valid ipv4 address starting from index i; false otherwise and if index i is invalide value

	int len = s.length();
	if (i <= 0 || i >= len) {
	    return false;
	}

	char c;
	int j;
	int initI = i;

	for (int num = 1; num < 5; num++) {

	    for (j = 0; j < 3 && i < len; i++, j++) {
		
		c = s.charAt(i);
		
		if (!digit(c)) {
		    break;
		}
	    }
	    
	    // there should be at least one digit and 
	    // the number between dots should be less then 256
	    if (j == 0 || 
		(j == 3 && 
		 (((s.charAt(i-3)-'0")*10 + (s.charAt(i-2)-'0"))*10 +
		  (s.charAt(i-1)-'0")) > 255)) {
		return false;
	    }
	    
	    // check if this is the end of the string
	    if (i == len)  {
		if (num == 4) {
		    mmsAddress.set(s.substring(initI), null, IPV4);
		    return true;
		}
		return false;
	    }

	    c = s.charAt(i);

	    if (c == ':") {
		mmsAddress.set(s.substring(initI, i), null, IPV4);
		return parseApplicationId(s, i, mmsAddress);
	    }

	    if (c != '.") {
		return false; // 4th character after beg (or dot)
		// should be dot;
		// only dots are allowed as non digit char
	    }

	    // allowed '.' => continue
	    i++;
	}
	
	return false;
    
static booleanparseIpv6(java.lang.String s, int i, com.sun.tck.wma.mms.MMSAddress mmsAddress)
Determines whether a string represents a valid ipv6 address starting from an index. If the address string is a valid ipv6 representation it is parsed into the fields of MMSAddress object (address and application id). ipv6 syntax is specified in WMA 2.0 spec as : ipv6-atom ":" ipv6-atom ":" ipv6-atom ":" ipv6-atom ":" ipv6-atom ":" ipv6-atom ":" ipv6-atom ":" ipv6-atom [appId] where ipv6-atom is 1*4(digit | hex-alpha).

param
s The string to check.
param
mmsAddress The return MMSAddress object which fields are filled with parsed address and application id (which can be null), its type is set to IPV6.
param
i The index of the string at which to start the check. 0 <= i <= s.length()
return
true if s represents a valid ipv6 address starting from index i; false otherwise and if index i is invalide value

	int len = s.length();
	if (i <= 0 || i >= len) {
	    return false;
	}

	char c;
	int j;
	int initI = i;

	for (int num = 1; num < 9; num++) {

	    for (j = 0; j < 4 && i < len; i++, j++) {
		
		c = s.charAt(i);
		
		if (!hex(c)) {
		    break;
		}
	    }
	    
	    // there should be at least one digit 
	    if (j == 0) {
		return false;
	    }

	    
	    // check if this is the end of the string
	    if (i == len)  {
		if (num == 8) {
		    mmsAddress.set(s.substring(initI), null, IPV6);
		    return true;
		}
		return false;
	    }

	    c = s.charAt(i);

	    if (c == ':") { 
		if (num == 8) {
		    mmsAddress.set(s.substring(initI, i), 
				   null, IPV6);
		    return parseApplicationId(s, i, mmsAddress);
		}
	    } else {
		return false; // 5th character after beg (or :)
		// should be :
		// only : are allowed as non digit char
	    }

	    // allowed ':' => continue
	    i++;
	}
	
	return false;
    
static booleanparsePhoneNumber(java.lang.String s, int i, com.sun.tck.wma.mms.MMSAddress mmsAddress)
Determines whether a string represents a valid phone number starting from an index. If the address string is a valid phone number representation it is parsed into the fields of MMSAddress object (address and application id). General phone number syntax is specified in WMA 2.0 spec as : ("+" 1*digit | 1*digit) [applicationId]

param
s The string to check.
param
mmsAddress The return MMSAddress object which fields are filled with parsed address and application id (which can be null), its type is set to GLOBAL_PHONE_NUMBER.
param
i The index of the string at which to start the check. 0 <= i <= s.length()
return
true if s represents a valid phone number starting from index i; false otherwise and if index i is invalide value


	int len = s.length();
	if (i < 0 || i >= len) {
	    return false;
	}

	int initI = i;
	char c = s.charAt(i);

	if (c == '+") {
	    i++;
	    if (len == i) {
		return false;
	    }
	}

	int j = 0;
	for (; i < len; j++, i++) {
	    c = s.charAt(i);
	    if (!digit(c)) {
		break;
	    }
	}
	
	if (j == 0) {
	    return false;
	}

	if (i == len) {
	    mmsAddress.set(s.substring(initI), null, GLOBAL_PHONE_NUMBER);
	    return true;
	}

	if (c == ':") {
	    mmsAddress.set(s.substring(initI, i), 
			   null, GLOBAL_PHONE_NUMBER);
	    return parseApplicationId(s, i, mmsAddress);
	}
	
	return false;
    
static booleanparseShortCode(java.lang.String s, int i, com.sun.tck.wma.mms.MMSAddress mmsAddress)
Determines whether a string represents a valid shortcode starting from an index. If the address string is a valid shortcode representation it is parsed into the fields of MMSAddress object (address and null for appId). Shortcode syntax is specified in WMA 2.0 spec as : *(digit | alpha)

param
s The string to check.
param
mmsAddress The return MMSAddress object which fields are filled with parsed address and application id (null in this case), its type is set to SHORTCODE.
param
i The index of the string at which to start the check. 0 <= i <= s.length()
return
true if s represents a valid shortcode starting from index i; false otherwise and if index i is invalide value

	int len = s.length();
	if (i > len) {
	    return false;
	}

	if (i == len) {
            return false; // Even though spec allows "mms://" as a valid 
            // shortcode, we disallow it here
	}
        int initI = i;
	char c;
	
	for (; i < s.length(); i++) {
	    c = s.charAt(i);

	    if (!digit(c) && !alpha(c)) {
		return false;
	    }
	}
	mmsAddress.set(s.substring(initI), null, SHORTCODE);
	return true;
    
voidset(java.lang.String address, java.lang.String appId, int type)
Sets MMSAddress fields to the passed in values.

param
address The address part of the mms address
param
appId The application id part of the mms address
param
type The type of this mms addreess (EMAIL, GLOBAL_PHONE_NUMBER, IPV4, IPV6, SHORTCODE, APP_ID)

	this.address = address;
	this.appId   = appId;
	this.type    = type;
    
voidsetAppid(java.lang.String appId)
Sets MMSAddress application id field to the passed in value.

param
appId The application id part of the mms address

	this.appId = appId; 
    
private static booleanspecials(char c)
Determines whether a character represents a special e-mail character.

param
c The character to check.
return
true if c is a special character. false otherwise

	if (c == '(" || c == ')" || c == '<" || c == '>" ||
	    c == '@" || c == '," || c == ';" || c == ':" ||
	    c == '\\" || c == '"" || c == '." || c == '[" || c == ']") {
	    return true;
	}
	return false;