panda.net.ftp.FTPClientConfig Maven / Gradle / Ivy
package panda.net.ftp;
import java.text.DateFormatSymbols;
import java.util.Collection;
import java.util.Locale;
import java.util.Map;
import java.util.StringTokenizer;
import java.util.TreeMap;
/**
*
* This class implements an alternate means of configuring the {@link panda.net.ftp.FTPClient
* FTPClient} object and also subordinate objects which it uses. Any class implementing the
* {@link panda.net.ftp.Configurable Configurable } interface can be configured by this object.
*
*
* In particular this class was designed primarily to support configuration of FTP servers which
* express file timestamps in formats and languages other than those for the US locale, which
* although it is the most common is not universal. Unfortunately, nothing in the FTP spec allows
* this to be determined in an automated way, so manual configuration such as this is necessary.
*
*
* This functionality was designed to allow existing clients to work exactly as before without
* requiring use of this component. This component should only need to be explicitly invoked by the
* user of this package for problem cases that previous implementations could not solve.
*
* Examples of use of FTPClientConfig
Use cases: You are trying to access a server that
*
* - lists files with timestamps that use month names in languages other than English
* - lists files with timestamps that use date formats other than the American English "standard"
*
MM dd yyyy
* - is in different timezone and you need accurate timestamps for dependency checking as in Ant
*
*
* Unpaged (whole list) access on a UNIX server that uses French month names but uses the "standard"
* MMM d yyyy
date formatting
*
*
* FTPClient f = FTPClient();
* FTPClientConfig conf = new FTPClientConfig(FTPClientConfig.SYST_UNIX);
* conf.setServerLanguageCode("fr");
* f.configure(conf);
* f.connect(server);
* f.login(username, password);
* FTPFile[] files = listFiles(directory);
*
*
* Paged access on a UNIX server that uses Danish month names and "European" date formatting in
* Denmark's time zone, when you are in some other time zone.
*
*
* FTPClient f = FTPClient();
* FTPClientConfig conf = new FTPClientConfig(FTPClientConfig.SYST_UNIX);
* conf.setServerLanguageCode("da");
* conf.setDefaultDateFormat("d MMM yyyy");
* conf.setRecentDateFormat("d MMM HH:mm");
* conf.setTimeZoneId("Europe/Copenhagen");
* f.configure(conf);
* f.connect(server);
* f.login(username, password);
* FTPListParseEngine engine = f.initiateListParsing("com.whatever.YourOwnParser", directory);
*
* while (engine.hasNext()) {
* FTPFile[] files = engine.getNext(25); // "page size" you want
* // do whatever you want with these files, display them, etc.
* // expensive FTPFile objects not created until needed.
* }
*
*
* Unpaged (whole list) access on a VMS server that uses month names in a language not
* {@link #getSupportedLanguageCodes() supported} by the system. but uses the "standard"
* MMM d yyyy
date formatting
*
*
* FTPClient f = FTPClient();
* FTPClientConfig conf = new FTPClientConfig(FTPClientConfig.SYST_VMS);
* conf.setShortMonthNames("jan|feb|mar|apr|ma\u00ED|j\u00FAn|j\u00FAl|\u00e1g\u00FA|sep|okt|n\u00F3v|des");
* f.configure(conf);
* f.connect(server);
* f.login(username, password);
* FTPFile[] files = listFiles(directory);
*
*
* Unpaged (whole list) access on a Windows-NT server in a different time zone. (Note, since the NT
* Format uses numeric date formatting, language issues are irrelevant here).
*
*
* FTPClient f = FTPClient();
* FTPClientConfig conf = new FTPClientConfig(FTPClientConfig.SYST_NT);
* conf.setTimeZoneId("America/Denver");
* f.configure(conf);
* f.connect(server);
* f.login(username, password);
* FTPFile[] files = listFiles(directory);
*
*
* Unpaged (whole list) access on a Windows-NT server in a different time zone but which has been
* configured to use a unix-style listing format.
*
*
* FTPClient f = FTPClient();
* FTPClientConfig conf = new FTPClientConfig(FTPClientConfig.SYST_UNIX);
* conf.setTimeZoneId("America/Denver");
* f.configure(conf);
* f.connect(server);
* f.login(username, password);
* FTPFile[] files = listFiles(directory);
*
*
* @see panda.net.ftp.Configurable
* @see panda.net.ftp.FTPClient
* @see panda.net.ftp.parser.FTPTimestampParserImpl#configure(FTPClientConfig)
* @see panda.net.ftp.parser.ConfigurableFTPFileEntryParserImpl
*/
public class FTPClientConfig {
/**
* Identifier by which a unix-based ftp server is known throughout the commons-net ftp system.
*/
public static final String SYST_UNIX = "UNIX";
/**
* Identifier for alternate UNIX parser; same as {@link #SYST_UNIX} but leading spaces are
* trimmed from file names. This is to maintain backwards compatibility with the original
* behaviour of the parser which ignored multiple spaces between the date and the start of the
* file name.
*
*/
public static final String SYST_UNIX_TRIM_LEADING = "UNIX_LTRIM";
/**
* Identifier by which a vms-based ftp server is known throughout the commons-net ftp system.
*/
public static final String SYST_VMS = "VMS";
/**
* Identifier by which a WindowsNT-based ftp server is known throughout the commons-net ftp
* system.
*/
public static final String SYST_NT = "WINDOWS";
/**
* Identifier by which an OS/2-based ftp server is known throughout the commons-net ftp system.
*/
public static final String SYST_OS2 = "OS/2";
/**
* Identifier by which an OS/400-based ftp server is known throughout the commons-net ftp
* system.
*/
public static final String SYST_OS400 = "OS/400";
/**
* Identifier by which an AS/400-based ftp server is known throughout the commons-net ftp
* system.
*/
public static final String SYST_AS400 = "AS/400";
/**
* Identifier by which an MVS-based ftp server is known throughout the commons-net ftp system.
*/
public static final String SYST_MVS = "MVS";
/**
* Some servers return an "UNKNOWN Type: L8" message in response to the SYST command. We set
* these to be a Unix-type system. This may happen if the ftpd in question was compiled without
* system information. NET-230 - Updated to be UPPERCASE so that the check done in
* createFileEntryParser will succeed.
*
*/
public static final String SYST_L8 = "TYPE: L8";
/**
* Identifier by which an Netware-based ftp server is known throughout the commons-net ftp
* system.
*
*/
public static final String SYST_NETWARE = "NETWARE";
/**
* Identifier by which a Mac pre OS-X -based ftp server is known throughout the commons-net ftp
* system.
*
*/
// Full string is "MACOS Peter's Server"; the substring below should be enough
public static final String SYST_MACOS_PETER = "MACOS PETER"; // NET-436
private final String serverSystemKey;
private String defaultDateFormatStr = null;
private String recentDateFormatStr = null;
private boolean lenientFutureDates = true; // NET-407
private String serverLanguageCode = null;
private String shortMonthNames = null;
private String serverTimeZoneId = null;
private boolean saveUnparseableEntries = false;
/**
* The main constructor for an FTPClientConfig object
*
* @param systemKey key representing system type of the server being connected to. See
* {@link #getServerSystemKey() serverSystemKey} If set to the empty string, then
* FTPClient uses the system type returned by the server. However this is not
* recommended for general use; the correct system type should be set if it is known.
*/
public FTPClientConfig(String systemKey) {
this.serverSystemKey = systemKey;
}
/**
* Convenience constructor mainly for use in testing. Constructs a UNIX configuration.
*/
public FTPClientConfig() {
this(SYST_UNIX);
}
/**
* Constructor which allows setting of most member fields
*
* @param systemKey key representing system type of the server being connected to. See
* {@link #getServerSystemKey() serverSystemKey}
* @param defaultDateFormatStr See {@link #setDefaultDateFormatStr(String) defaultDateFormatStr}
* @param recentDateFormatStr See {@link #setRecentDateFormatStr(String) recentDateFormatStr}
* @param serverLanguageCode See {@link #setServerLanguageCode(String) serverLanguageCode}
* @param shortMonthNames See {@link #setShortMonthNames(String) shortMonthNames}
* @param serverTimeZoneId See {@link #setServerTimeZoneId(String) serverTimeZoneId}
*/
public FTPClientConfig(String systemKey, String defaultDateFormatStr, String recentDateFormatStr,
String serverLanguageCode, String shortMonthNames, String serverTimeZoneId) {
this(systemKey);
this.defaultDateFormatStr = defaultDateFormatStr;
this.recentDateFormatStr = recentDateFormatStr;
this.serverLanguageCode = serverLanguageCode;
this.shortMonthNames = shortMonthNames;
this.serverTimeZoneId = serverTimeZoneId;
}
/**
* Constructor which allows setting of all member fields
*
* @param systemKey key representing system type of the server being connected to. See
* {@link #getServerSystemKey() serverSystemKey}
* @param defaultDateFormatStr See {@link #setDefaultDateFormatStr(String) defaultDateFormatStr}
* @param recentDateFormatStr See {@link #setRecentDateFormatStr(String) recentDateFormatStr}
* @param serverLanguageCode See {@link #setServerLanguageCode(String) serverLanguageCode}
* @param shortMonthNames See {@link #setShortMonthNames(String) shortMonthNames}
* @param serverTimeZoneId See {@link #setServerTimeZoneId(String) serverTimeZoneId}
* @param lenientFutureDates See {@link #setLenientFutureDates(boolean) lenientFutureDates}
* @param saveUnparseableEntries See {@link #setUnparseableEntries(boolean)
* saveUnparseableEntries}
*/
public FTPClientConfig(String systemKey, String defaultDateFormatStr, String recentDateFormatStr,
String serverLanguageCode, String shortMonthNames, String serverTimeZoneId, boolean lenientFutureDates,
boolean saveUnparseableEntries) {
this(systemKey);
this.defaultDateFormatStr = defaultDateFormatStr;
this.lenientFutureDates = lenientFutureDates;
this.recentDateFormatStr = recentDateFormatStr;
this.saveUnparseableEntries = saveUnparseableEntries;
this.serverLanguageCode = serverLanguageCode;
this.shortMonthNames = shortMonthNames;
this.serverTimeZoneId = serverTimeZoneId;
}
// Copy constructor, intended for use by FTPClient only
FTPClientConfig(String systemKey, FTPClientConfig config) {
this.serverSystemKey = systemKey;
this.defaultDateFormatStr = config.defaultDateFormatStr;
this.lenientFutureDates = config.lenientFutureDates;
this.recentDateFormatStr = config.recentDateFormatStr;
this.saveUnparseableEntries = config.saveUnparseableEntries;
this.serverLanguageCode = config.serverLanguageCode;
this.serverTimeZoneId = config.serverTimeZoneId;
this.shortMonthNames = config.shortMonthNames;
}
private static final Map LANGUAGE_CODE_MAP = new TreeMap();
static {
// if there are other commonly used month name encodings which
// correspond to particular locales, please add them here.
// many locales code short names for months as all three letters
// these we handle simply.
LANGUAGE_CODE_MAP.put("en", Locale.ENGLISH);
LANGUAGE_CODE_MAP.put("de", Locale.GERMAN);
LANGUAGE_CODE_MAP.put("it", Locale.ITALIAN);
LANGUAGE_CODE_MAP.put("es", new Locale("es", "", "")); // spanish
LANGUAGE_CODE_MAP.put("pt", new Locale("pt", "", "")); // portuguese
LANGUAGE_CODE_MAP.put("da", new Locale("da", "", "")); // danish
LANGUAGE_CODE_MAP.put("sv", new Locale("sv", "", "")); // swedish
LANGUAGE_CODE_MAP.put("no", new Locale("no", "", "")); // norwegian
LANGUAGE_CODE_MAP.put("nl", new Locale("nl", "", "")); // dutch
LANGUAGE_CODE_MAP.put("ro", new Locale("ro", "", "")); // romanian
LANGUAGE_CODE_MAP.put("sq", new Locale("sq", "", "")); // albanian
LANGUAGE_CODE_MAP.put("sh", new Locale("sh", "", "")); // serbo-croatian
LANGUAGE_CODE_MAP.put("sk", new Locale("sk", "", "")); // slovak
LANGUAGE_CODE_MAP.put("sl", new Locale("sl", "", "")); // slovenian
// some don't
LANGUAGE_CODE_MAP.put("fr", "jan|f\u00e9v|mar|avr|mai|jun|jui|ao\u00fb|sep|oct|nov|d\u00e9c"); // french
}
/**
* Getter for the serverSystemKey property. This property specifies the general type of server
* to which the client connects. Should be either one of the FTPClientConfig.SYST_*
* codes or else the fully qualified class name of a parser implementing both the
* FTPFileEntryParser
and Configurable
interfaces.
*
* @return Returns the serverSystemKey property.
*/
public String getServerSystemKey() {
return serverSystemKey;
}
/**
* getter for the {@link #setDefaultDateFormatStr(String) defaultDateFormatStr} property.
*
* @return Returns the defaultDateFormatStr property.
*/
public String getDefaultDateFormatStr() {
return defaultDateFormatStr;
}
/**
* getter for the {@link #setRecentDateFormatStr(String) recentDateFormatStr} property.
*
* @return Returns the recentDateFormatStr property.
*/
public String getRecentDateFormatStr() {
return recentDateFormatStr;
}
/**
* getter for the {@link #setServerTimeZoneId(String) serverTimeZoneId} property.
*
* @return Returns the serverTimeZoneId property.
*/
public String getServerTimeZoneId() {
return serverTimeZoneId;
}
/**
*
* getter for the {@link #setShortMonthNames(String) shortMonthNames} property.
*
*
* @return Returns the shortMonthNames.
*/
public String getShortMonthNames() {
return shortMonthNames;
}
/**
*
* getter for the {@link #setServerLanguageCode(String) serverLanguageCode} property.
*
*
* @return Returns the serverLanguageCode property.
*/
public String getServerLanguageCode() {
return serverLanguageCode;
}
/**
*
* getter for the {@link #setLenientFutureDates(boolean) lenientFutureDates} property.
*
*
* @return Returns the lenientFutureDates.
*/
public boolean isLenientFutureDates() {
return lenientFutureDates;
}
/**
*
* setter for the defaultDateFormatStr property. This property specifies the main date format
* that will be used by a parser configured by this configuration to parse file timestamps. If
* this is not specified, such a parser will use as a default value, the most commonly used
* format which will be in as used in en_US
locales.
*
*
* This should be in the format described for java.text.SimpleDateFormat
. property.
*
*
* @param defaultDateFormatStr The defaultDateFormatStr to set.
*/
public void setDefaultDateFormatStr(String defaultDateFormatStr) {
this.defaultDateFormatStr = defaultDateFormatStr;
}
/**
*
* setter for the recentDateFormatStr property. This property specifies a secondary date format
* that will be used by a parser configured by this configuration to parse file timestamps,
* typically those less than a year old. If this is not specified, such a parser will not
* attempt to parse using an alternate format.
*
*
* This is used primarily in unix-based systems.
*
*
* This should be in the format described for java.text.SimpleDateFormat
.
*
*
* @param recentDateFormatStr The recentDateFormatStr to set.
*/
public void setRecentDateFormatStr(String recentDateFormatStr) {
this.recentDateFormatStr = recentDateFormatStr;
}
/**
*
* setter for the lenientFutureDates property. This boolean property (default: false) only has
* meaning when a {@link #setRecentDateFormatStr(String) recentDateFormatStr} property has been
* set. In that case, if this property is set true, then the parser, when it encounters a
* listing parseable with the recent date format, will only consider a date to belong to the
* previous year if it is more than one day in the future. This will allow all out-of-synch
* situations (whether based on "slop" - i.e. servers simply out of synch with one another or
* because of time zone differences - but in the latter case it is highly recommended to use the
* {@link #setServerTimeZoneId(String) serverTimeZoneId} property instead) to resolve correctly.
*
*
* This is used primarily in unix-based systems.
*
*
* @param lenientFutureDates set true to compensate for out-of-synch conditions.
*/
public void setLenientFutureDates(boolean lenientFutureDates) {
this.lenientFutureDates = lenientFutureDates;
}
/**
*
* setter for the serverTimeZoneId property. This property allows a time zone to be specified
* corresponding to that known to be used by an FTP server in file listings. This might be
* particularly useful to clients such as Ant that try to use these timestamps for dependency
* checking.
*
*
* This should be one of the identifiers used by java.util.TimeZone
to refer to
* time zones, for example, America/Chicago
or Asia/Rangoon
.
*
*
* @param serverTimeZoneId The serverTimeZoneId to set.
*/
public void setServerTimeZoneId(String serverTimeZoneId) {
this.serverTimeZoneId = serverTimeZoneId;
}
/**
*
* setter for the shortMonthNames property. This property allows the user to specify a set of
* month names used by the server that is different from those that may be specified using the
* {@link #setServerLanguageCode(String) serverLanguageCode} property.
*
*
* This should be a string containing twelve strings each composed of three characters,
* delimited by pipe (|) characters. Currently, only 8-bit ASCII characters are known to be
* supported. For example, a set of month names used by a hypothetical Icelandic FTP server
* might conceivably be specified as
* "jan|feb|mar|apr|maí|jún|júl|ágú|sep|okt|nóv|des"
.
*
*
* @param shortMonthNames The value to set to the shortMonthNames property.
*/
public void setShortMonthNames(String shortMonthNames) {
this.shortMonthNames = shortMonthNames;
}
/**
*
* setter for the serverLanguageCode property. This property allows user to specify a two-letter ISO-639 language
* code that will be used to configure the set of month names used by the file timestamp
* parser. If neither this nor the {@link #setShortMonthNames(String) shortMonthNames} is
* specified, parsing will assume English month names, which may or may not be significant,
* depending on whether the date format(s) specified via
* {@link #setDefaultDateFormatStr(String) defaultDateFormatStr} and/or
* {@link #setRecentDateFormatStr(String) recentDateFormatStr} are using numeric or alphabetic
* month names.
*
*
* If the code supplied is not supported here, en_US
month names will be used. We
* are supporting here those language codes which, when a java.util.Locale
is
* constucted using it, and a java.text.SimpleDateFormat
is constructed using that
* Locale, the array returned by the SimpleDateFormat's getShortMonths()
method
* consists solely of three 8-bit ASCII character strings. Additionally, languages which do not
* meet this requirement are included if a common alternative set of short month names is known
* to be used. This means that users who can tell us of additional such encodings may get them
* added to the list of supported languages by contacting the Apache Commons Net team.
*
*
* Please note that this attribute will NOT be used to determine a locale-based date
* format for the language. Experience has shown that many if not most FTP servers
* outside the United States employ the standard en_US
date format orderings of
* MMM d yyyy
and MMM d HH:mm
and attempting to deduce this
* automatically here would cause more problems than it would solve. The date format must be
* changed via the {@link #setDefaultDateFormatStr(String) defaultDateFormatStr} and/or
* {@link #setRecentDateFormatStr(String) recentDateFormatStr} parameters.
*
*
* @param serverLanguageCode The value to set to the serverLanguageCode property.
*/
public void setServerLanguageCode(String serverLanguageCode) {
this.serverLanguageCode = serverLanguageCode;
}
/**
* Looks up the supplied language code in the internally maintained table of language codes.
* Returns a DateFormatSymbols object configured with short month names corresponding to the
* code. If there is no corresponding entry in the table, the object returned will be that for
* Locale.US
*
* @param languageCode See {@link #setServerLanguageCode(String) serverLanguageCode}
* @return a DateFormatSymbols object configured with short month names corresponding to the
* supplied code, or with month names for Locale.US
if there is no
* corresponding entry in the internal table.
*/
public static DateFormatSymbols lookupDateFormatSymbols(String languageCode) {
Object lang = LANGUAGE_CODE_MAP.get(languageCode);
if (lang != null) {
if (lang instanceof Locale) {
return new DateFormatSymbols((Locale)lang);
}
else if (lang instanceof String) {
return getDateFormatSymbols((String)lang);
}
}
return new DateFormatSymbols(Locale.US);
}
/**
* Returns a DateFormatSymbols object configured with short month names as in the supplied
* string
*
* @param shortmonths This should be as described in {@link #setShortMonthNames(String)
* shortMonthNames}
* @return a DateFormatSymbols object configured with short month names as in the supplied
* string
*/
public static DateFormatSymbols getDateFormatSymbols(String shortmonths) {
String[] months = splitShortMonthString(shortmonths);
DateFormatSymbols dfs = new DateFormatSymbols(Locale.US);
dfs.setShortMonths(months);
return dfs;
}
private static String[] splitShortMonthString(String shortmonths) {
StringTokenizer st = new StringTokenizer(shortmonths, "|");
int monthcnt = st.countTokens();
if (12 != monthcnt) {
throw new IllegalArgumentException("expecting a pipe-delimited string containing 12 tokens");
}
String[] months = new String[13];
int pos = 0;
while (st.hasMoreTokens()) {
months[pos++] = st.nextToken();
}
months[pos] = "";
return months;
}
/**
* Returns a Collection of all the language codes currently supported by this class. See
* {@link #setServerLanguageCode(String) serverLanguageCode} for a functional descrption of
* language codes within this system.
*
* @return a Collection of all the language codes currently supported by this class
*/
public static Collection getSupportedLanguageCodes() {
return LANGUAGE_CODE_MAP.keySet();
}
/**
* Allow list parsing methods to create basic FTPFile entries if parsing fails.
*
* In this case, the FTPFile will contain only the unparsed entry
* {@link FTPFile#getRawListing()} and {@link FTPFile#isValid()} will return {@code false}
*
* @param saveUnparseable if true, then create FTPFile entries if parsing fails
*/
public void setUnparseableEntries(boolean saveUnparseable) {
this.saveUnparseableEntries = saveUnparseable;
}
/**
* @return true if list parsing should return FTPFile entries even for unparseable response
* lines
*
* If true, the FTPFile for any unparseable entries will contain only the unparsed entry
* {@link FTPFile#getRawListing()} and {@link FTPFile#isValid()} will return
* {@code false}
*/
public boolean getUnparseableEntries() {
return this.saveUnparseableEntries;
}
}