Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

ExcelExtractor.java 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. /* ====================================================================
  2. Licensed to the Apache Software Foundation (ASF) under one or more
  3. contributor license agreements. See the NOTICE file distributed with
  4. this work for additional information regarding copyright ownership.
  5. The ASF licenses this file to You under the Apache License, Version 2.0
  6. (the "License"); you may not use this file except in compliance with
  7. the License. You may obtain a copy of the License at
  8. http://www.apache.org/licenses/LICENSE-2.0
  9. Unless required by applicable law or agreed to in writing, software
  10. distributed under the License is distributed on an "AS IS" BASIS,
  11. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. See the License for the specific language governing permissions and
  13. limitations under the License.
  14. ==================================================================== */
  15. package org.apache.poi.hssf.extractor;
  16. import java.io.File;
  17. import java.io.FileInputStream;
  18. import java.io.IOException;
  19. import java.io.InputStream;
  20. import java.io.PrintStream;
  21. import java.util.Locale;
  22. import org.apache.poi.POIOLE2TextExtractor;
  23. import org.apache.poi.hssf.usermodel.HSSFCell;
  24. import org.apache.poi.hssf.usermodel.HSSFCellStyle;
  25. import org.apache.poi.hssf.usermodel.HSSFComment;
  26. import org.apache.poi.hssf.usermodel.HSSFDataFormatter;
  27. import org.apache.poi.hssf.usermodel.HSSFRichTextString;
  28. import org.apache.poi.hssf.usermodel.HSSFRow;
  29. import org.apache.poi.hssf.usermodel.HSSFSheet;
  30. import org.apache.poi.hssf.usermodel.HSSFWorkbook;
  31. import org.apache.poi.poifs.filesystem.DirectoryNode;
  32. import org.apache.poi.poifs.filesystem.POIFSFileSystem;
  33. import org.apache.poi.ss.formula.eval.ErrorEval;
  34. import org.apache.poi.ss.usermodel.HeaderFooter;
  35. /**
  36. * A text extractor for Excel files.
  37. * <p>
  38. * Returns the textual content of the file, suitable for
  39. * indexing by something like Lucene, but not really
  40. * intended for display to the user.
  41. * </p>
  42. * <p>
  43. * To turn an excel file into a CSV or similar, then see
  44. * the XLS2CSVmra example
  45. * </p>
  46. *
  47. * @see <a href="http://svn.apache.org/repos/asf/poi/trunk/src/examples/src/org/apache/poi/hssf/eventusermodel/examples/XLS2CSVmra.java">XLS2CSVmra</a>
  48. */
  49. public class ExcelExtractor extends POIOLE2TextExtractor implements org.apache.poi.ss.extractor.ExcelExtractor {
  50. private HSSFWorkbook _wb;
  51. private HSSFDataFormatter _formatter;
  52. private boolean _includeSheetNames = true;
  53. private boolean _shouldEvaluateFormulas = true;
  54. private boolean _includeCellComments = false;
  55. private boolean _includeBlankCells = false;
  56. private boolean _includeHeadersFooters = true;
  57. public ExcelExtractor(HSSFWorkbook wb) {
  58. super(wb);
  59. _wb = wb;
  60. _formatter = new HSSFDataFormatter();
  61. }
  62. public ExcelExtractor(POIFSFileSystem fs) throws IOException {
  63. this(fs.getRoot());
  64. }
  65. public ExcelExtractor(DirectoryNode dir) throws IOException {
  66. this(new HSSFWorkbook(dir, true));
  67. }
  68. private static final class CommandParseException extends Exception {
  69. public CommandParseException(String msg) {
  70. super(msg);
  71. }
  72. }
  73. private static final class CommandArgs {
  74. private final boolean _requestHelp;
  75. private final File _inputFile;
  76. private final boolean _showSheetNames;
  77. private final boolean _evaluateFormulas;
  78. private final boolean _showCellComments;
  79. private final boolean _showBlankCells;
  80. private final boolean _headersFooters;
  81. public CommandArgs(String[] args) throws CommandParseException {
  82. int nArgs = args.length;
  83. File inputFile = null;
  84. boolean requestHelp = false;
  85. boolean showSheetNames = true;
  86. boolean evaluateFormulas = true;
  87. boolean showCellComments = false;
  88. boolean showBlankCells = false;
  89. boolean headersFooters = true;
  90. for (int i=0; i<nArgs; i++) {
  91. String arg = args[i];
  92. if ("-help".equalsIgnoreCase(arg)) {
  93. requestHelp = true;
  94. break;
  95. }
  96. if ("-i".equals(arg)) {
  97. // step to next arg
  98. if (++i >= nArgs) {
  99. throw new CommandParseException("Expected filename after '-i'");
  100. }
  101. arg = args[i];
  102. if (inputFile != null) {
  103. throw new CommandParseException("Only one input file can be supplied");
  104. }
  105. inputFile = new File(arg);
  106. if (!inputFile.exists()) {
  107. throw new CommandParseException("Specified input file '" + arg + "' does not exist");
  108. }
  109. if (inputFile.isDirectory()) {
  110. throw new CommandParseException("Specified input file '" + arg + "' is a directory");
  111. }
  112. continue;
  113. }
  114. if ("--show-sheet-names".equals(arg)) {
  115. showSheetNames = parseBoolArg(args, ++i);
  116. continue;
  117. }
  118. if ("--evaluate-formulas".equals(arg)) {
  119. evaluateFormulas = parseBoolArg(args, ++i);
  120. continue;
  121. }
  122. if ("--show-comments".equals(arg)) {
  123. showCellComments = parseBoolArg(args, ++i);
  124. continue;
  125. }
  126. if ("--show-blanks".equals(arg)) {
  127. showBlankCells = parseBoolArg(args, ++i);
  128. continue;
  129. }
  130. if ("--headers-footers".equals(arg)) {
  131. headersFooters = parseBoolArg(args, ++i);
  132. continue;
  133. }
  134. throw new CommandParseException("Invalid argument '" + arg + "'");
  135. }
  136. _requestHelp = requestHelp;
  137. _inputFile = inputFile;
  138. _showSheetNames = showSheetNames;
  139. _evaluateFormulas = evaluateFormulas;
  140. _showCellComments = showCellComments;
  141. _showBlankCells = showBlankCells;
  142. _headersFooters = headersFooters;
  143. }
  144. private static boolean parseBoolArg(String[] args, int i) throws CommandParseException {
  145. if (i >= args.length) {
  146. throw new CommandParseException("Expected value after '" + args[i-1] + "'");
  147. }
  148. String value = args[i].toUpperCase(Locale.ROOT);
  149. if ("Y".equals(value) || "YES".equals(value) || "ON".equals(value) || "TRUE".equals(value)) {
  150. return true;
  151. }
  152. if ("N".equals(value) || "NO".equals(value) || "OFF".equals(value) || "FALSE".equals(value)) {
  153. return false;
  154. }
  155. throw new CommandParseException("Invalid value '" + args[i] + "' for '" + args[i-1] + "'. Expected 'Y' or 'N'");
  156. }
  157. public boolean isRequestHelp() {
  158. return _requestHelp;
  159. }
  160. public File getInputFile() {
  161. return _inputFile;
  162. }
  163. public boolean shouldShowSheetNames() {
  164. return _showSheetNames;
  165. }
  166. public boolean shouldEvaluateFormulas() {
  167. return _evaluateFormulas;
  168. }
  169. public boolean shouldShowCellComments() {
  170. return _showCellComments;
  171. }
  172. public boolean shouldShowBlankCells() {
  173. return _showBlankCells;
  174. }
  175. public boolean shouldIncludeHeadersFooters() {
  176. return _headersFooters;
  177. }
  178. }
  179. private static void printUsageMessage(PrintStream ps) {
  180. ps.println("Use:");
  181. ps.println(" " + ExcelExtractor.class.getName() + " [<flag> <value> [<flag> <value> [...]]] [-i <filename.xls>]");
  182. ps.println(" -i <filename.xls> specifies input file (default is to use stdin)");
  183. ps.println(" Flags can be set on or off by using the values 'Y' or 'N'.");
  184. ps.println(" Following are available flags and their default values:");
  185. ps.println(" --show-sheet-names Y");
  186. ps.println(" --evaluate-formulas Y");
  187. ps.println(" --show-comments N");
  188. ps.println(" --show-blanks Y");
  189. ps.println(" --headers-footers Y");
  190. }
  191. /**
  192. * Command line extractor.
  193. * @throws IOException
  194. */
  195. public static void main(String[] args) throws IOException {
  196. CommandArgs cmdArgs;
  197. try {
  198. cmdArgs = new CommandArgs(args);
  199. } catch (CommandParseException e) {
  200. System.err.println(e.getMessage());
  201. printUsageMessage(System.err);
  202. System.exit(1);
  203. return; // suppress compiler error
  204. }
  205. if (cmdArgs.isRequestHelp()) {
  206. printUsageMessage(System.out);
  207. return;
  208. }
  209. InputStream is;
  210. if(cmdArgs.getInputFile() == null) {
  211. is = System.in;
  212. } else {
  213. is = new FileInputStream(cmdArgs.getInputFile());
  214. }
  215. HSSFWorkbook wb = new HSSFWorkbook(is);
  216. is.close();
  217. ExcelExtractor extractor = new ExcelExtractor(wb);
  218. extractor.setIncludeSheetNames(cmdArgs.shouldShowSheetNames());
  219. extractor.setFormulasNotResults(!cmdArgs.shouldEvaluateFormulas());
  220. extractor.setIncludeCellComments(cmdArgs.shouldShowCellComments());
  221. extractor.setIncludeBlankCells(cmdArgs.shouldShowBlankCells());
  222. extractor.setIncludeHeadersFooters(cmdArgs.shouldIncludeHeadersFooters());
  223. System.out.println(extractor.getText());
  224. extractor.close();
  225. }
  226. /**
  227. * Should sheet names be included? Default is true
  228. */
  229. public void setIncludeSheetNames(boolean includeSheetNames) {
  230. _includeSheetNames = includeSheetNames;
  231. }
  232. /**
  233. * Should we return the formula itself, and not
  234. * the result it produces? Default is false
  235. */
  236. public void setFormulasNotResults(boolean formulasNotResults) {
  237. _shouldEvaluateFormulas = !formulasNotResults;
  238. }
  239. /**
  240. * Should cell comments be included? Default is false
  241. */
  242. public void setIncludeCellComments(boolean includeCellComments) {
  243. _includeCellComments = includeCellComments;
  244. }
  245. /**
  246. * Should blank cells be output? Default is to only
  247. * output cells that are present in the file and are
  248. * non-blank.
  249. */
  250. public void setIncludeBlankCells(boolean includeBlankCells) {
  251. _includeBlankCells = includeBlankCells;
  252. }
  253. /**
  254. * Should headers and footers be included in the output?
  255. * Default is to include them.
  256. */
  257. public void setIncludeHeadersFooters(boolean includeHeadersFooters) {
  258. _includeHeadersFooters = includeHeadersFooters;
  259. }
  260. /**
  261. * Retrieves the text contents of the file
  262. */
  263. public String getText() {
  264. StringBuffer text = new StringBuffer();
  265. // We don't care about the difference between
  266. // null (missing) and blank cells
  267. _wb.setMissingCellPolicy(HSSFRow.RETURN_BLANK_AS_NULL);
  268. // Process each sheet in turn
  269. for(int i=0;i<_wb.getNumberOfSheets();i++) {
  270. HSSFSheet sheet = _wb.getSheetAt(i);
  271. if(sheet == null) { continue; }
  272. if(_includeSheetNames) {
  273. String name = _wb.getSheetName(i);
  274. if(name != null) {
  275. text.append(name);
  276. text.append("\n");
  277. }
  278. }
  279. // Header text, if there is any
  280. if(_includeHeadersFooters) {
  281. text.append(_extractHeaderFooter(sheet.getHeader()));
  282. }
  283. int firstRow = sheet.getFirstRowNum();
  284. int lastRow = sheet.getLastRowNum();
  285. for(int j=firstRow;j<=lastRow;j++) {
  286. HSSFRow row = sheet.getRow(j);
  287. if(row == null) { continue; }
  288. // Check each cell in turn
  289. int firstCell = row.getFirstCellNum();
  290. int lastCell = row.getLastCellNum();
  291. if(_includeBlankCells) {
  292. firstCell = 0;
  293. }
  294. for(int k=firstCell;k<lastCell;k++) {
  295. HSSFCell cell = row.getCell(k);
  296. boolean outputContents = true;
  297. if(cell == null) {
  298. // Only output if requested
  299. outputContents = _includeBlankCells;
  300. } else {
  301. switch(cell.getCellType()) {
  302. case HSSFCell.CELL_TYPE_STRING:
  303. text.append(cell.getRichStringCellValue().getString());
  304. break;
  305. case HSSFCell.CELL_TYPE_NUMERIC:
  306. text.append(
  307. _formatter.formatCellValue(cell)
  308. );
  309. break;
  310. case HSSFCell.CELL_TYPE_BOOLEAN:
  311. text.append(cell.getBooleanCellValue());
  312. break;
  313. case HSSFCell.CELL_TYPE_ERROR:
  314. text.append(ErrorEval.getText(cell.getErrorCellValue()));
  315. break;
  316. case HSSFCell.CELL_TYPE_FORMULA:
  317. if(!_shouldEvaluateFormulas) {
  318. text.append(cell.getCellFormula());
  319. } else {
  320. switch(cell.getCachedFormulaResultType()) {
  321. case HSSFCell.CELL_TYPE_STRING:
  322. HSSFRichTextString str = cell.getRichStringCellValue();
  323. if(str != null && str.length() > 0) {
  324. text.append(str.toString());
  325. }
  326. break;
  327. case HSSFCell.CELL_TYPE_NUMERIC:
  328. HSSFCellStyle style = cell.getCellStyle();
  329. double nVal = cell.getNumericCellValue();
  330. short df = style.getDataFormat();
  331. String dfs = style.getDataFormatString();
  332. text.append(_formatter.formatRawCellContents(nVal, df, dfs));
  333. break;
  334. case HSSFCell.CELL_TYPE_BOOLEAN:
  335. text.append(cell.getBooleanCellValue());
  336. break;
  337. case HSSFCell.CELL_TYPE_ERROR:
  338. text.append(ErrorEval.getText(cell.getErrorCellValue()));
  339. break;
  340. default:
  341. throw new IllegalStateException("Unexpected cell cached formula result type: " + cell.getCachedFormulaResultType());
  342. }
  343. }
  344. break;
  345. default:
  346. throw new RuntimeException("Unexpected cell type (" + cell.getCellType() + ")");
  347. }
  348. // Output the comment, if requested and exists
  349. HSSFComment comment = cell.getCellComment();
  350. if(_includeCellComments && comment != null) {
  351. // Replace any newlines with spaces, otherwise it
  352. // breaks the output
  353. String commentText = comment.getString().getString().replace('\n', ' ');
  354. text.append(" Comment by "+comment.getAuthor()+": "+commentText);
  355. }
  356. }
  357. // Output a tab if we're not on the last cell
  358. if(outputContents && k < (lastCell-1)) {
  359. text.append("\t");
  360. }
  361. }
  362. // Finish off the row
  363. text.append("\n");
  364. }
  365. // Finally Footer text, if there is any
  366. if(_includeHeadersFooters) {
  367. text.append(_extractHeaderFooter(sheet.getFooter()));
  368. }
  369. }
  370. return text.toString();
  371. }
  372. public static String _extractHeaderFooter(HeaderFooter hf) {
  373. StringBuffer text = new StringBuffer();
  374. if(hf.getLeft() != null) {
  375. text.append(hf.getLeft());
  376. }
  377. if(hf.getCenter() != null) {
  378. if(text.length() > 0)
  379. text.append("\t");
  380. text.append(hf.getCenter());
  381. }
  382. if(hf.getRight() != null) {
  383. if(text.length() > 0)
  384. text.append("\t");
  385. text.append(hf.getRight());
  386. }
  387. if(text.length() > 0)
  388. text.append("\n");
  389. return text.toString();
  390. }
  391. }