jbarnette-johnson 1.0.0.200806240111 → 1.0.0.200807291507

Sign up to get free protection for your applications and to get access to all the features.
Files changed (269) hide show
  1. data/MANIFEST +1 -0
  2. data/Rakefile +3 -10
  3. data/bin/johnson +2 -1
  4. data/ext/spidermonkey/context.c +3 -4
  5. data/ext/spidermonkey/context.h +1 -1
  6. data/ext/spidermonkey/conversions.c +39 -33
  7. data/ext/spidermonkey/debugger.c +5 -5
  8. data/ext/spidermonkey/immutable_node.c.erb +11 -11
  9. data/ext/spidermonkey/jroot.h +4 -4
  10. data/ext/spidermonkey/js_land_proxy.c +9 -8
  11. data/ext/spidermonkey/ruby_land_proxy.c +5 -4
  12. data/ext/spidermonkey/runtime.c +1 -1
  13. data/johnson.gemspec +36 -0
  14. data/lib/hoe.rb +0 -7
  15. data/lib/johnson/cli/options.rb +10 -4
  16. data/lib/johnson/spidermonkey/runtime.rb +2 -2
  17. data/lib/johnson/version.rb +4 -2
  18. data/lib/johnson.rb +1 -0
  19. data/test/johnson/runtime_test.rb +11 -0
  20. data/test/johnson/spidermonkey/ruby_land_proxy_test.rb +6 -0
  21. data/vendor/spidermonkey/.cvsignore +9 -0
  22. data/vendor/spidermonkey/Makefile.in +462 -0
  23. data/vendor/spidermonkey/Makefile.ref +364 -0
  24. data/vendor/spidermonkey/README.html +820 -0
  25. data/vendor/spidermonkey/SpiderMonkey.rsp +12 -0
  26. data/vendor/spidermonkey/Y.js +19 -0
  27. data/vendor/spidermonkey/build.mk +43 -0
  28. data/vendor/spidermonkey/config/AIX4.1.mk +65 -0
  29. data/vendor/spidermonkey/config/AIX4.2.mk +64 -0
  30. data/vendor/spidermonkey/config/AIX4.3.mk +65 -0
  31. data/vendor/spidermonkey/config/Darwin.mk +83 -0
  32. data/vendor/spidermonkey/config/Darwin1.3.mk +81 -0
  33. data/vendor/spidermonkey/config/Darwin1.4.mk +41 -0
  34. data/vendor/spidermonkey/config/Darwin5.2.mk +81 -0
  35. data/vendor/spidermonkey/config/Darwin5.3.mk +81 -0
  36. data/vendor/spidermonkey/config/HP-UXB.10.10.mk +77 -0
  37. data/vendor/spidermonkey/config/HP-UXB.10.20.mk +77 -0
  38. data/vendor/spidermonkey/config/HP-UXB.11.00.mk +80 -0
  39. data/vendor/spidermonkey/config/IRIX.mk +87 -0
  40. data/vendor/spidermonkey/config/IRIX5.3.mk +44 -0
  41. data/vendor/spidermonkey/config/IRIX6.1.mk +44 -0
  42. data/vendor/spidermonkey/config/IRIX6.2.mk +44 -0
  43. data/vendor/spidermonkey/config/IRIX6.3.mk +44 -0
  44. data/vendor/spidermonkey/config/IRIX6.5.mk +44 -0
  45. data/vendor/spidermonkey/config/Linux_All.mk +103 -0
  46. data/vendor/spidermonkey/config/Mac_OS10.0.mk +82 -0
  47. data/vendor/spidermonkey/config/OSF1V4.0.mk +72 -0
  48. data/vendor/spidermonkey/config/OSF1V5.0.mk +69 -0
  49. data/vendor/spidermonkey/config/SunOS4.1.4.mk +101 -0
  50. data/vendor/spidermonkey/config/SunOS5.10.mk +50 -0
  51. data/vendor/spidermonkey/config/SunOS5.3.mk +91 -0
  52. data/vendor/spidermonkey/config/SunOS5.4.mk +92 -0
  53. data/vendor/spidermonkey/config/SunOS5.5.1.mk +44 -0
  54. data/vendor/spidermonkey/config/SunOS5.5.mk +87 -0
  55. data/vendor/spidermonkey/config/SunOS5.6.mk +89 -0
  56. data/vendor/spidermonkey/config/SunOS5.7.mk +44 -0
  57. data/vendor/spidermonkey/config/SunOS5.8.mk +44 -0
  58. data/vendor/spidermonkey/config/SunOS5.9.mk +44 -0
  59. data/vendor/spidermonkey/config/WINNT4.0.mk +117 -0
  60. data/vendor/spidermonkey/config/WINNT5.0.mk +117 -0
  61. data/vendor/spidermonkey/config/WINNT5.1.mk +117 -0
  62. data/vendor/spidermonkey/config/WINNT5.2.mk +117 -0
  63. data/vendor/spidermonkey/config/WINNT6.0.mk +117 -0
  64. data/vendor/spidermonkey/config/dgux.mk +64 -0
  65. data/vendor/spidermonkey/config.mk +192 -0
  66. data/vendor/spidermonkey/editline/Makefile.ref +144 -0
  67. data/vendor/spidermonkey/editline/README +83 -0
  68. data/vendor/spidermonkey/editline/editline.3 +175 -0
  69. data/vendor/spidermonkey/editline/editline.c +1369 -0
  70. data/vendor/spidermonkey/editline/editline.h +135 -0
  71. data/vendor/spidermonkey/editline/sysunix.c +182 -0
  72. data/vendor/spidermonkey/editline/unix.h +82 -0
  73. data/vendor/spidermonkey/fdlibm/.cvsignore +7 -0
  74. data/vendor/spidermonkey/fdlibm/Makefile.in +127 -0
  75. data/vendor/spidermonkey/fdlibm/Makefile.ref +192 -0
  76. data/vendor/spidermonkey/fdlibm/e_acos.c +147 -0
  77. data/vendor/spidermonkey/fdlibm/e_acosh.c +105 -0
  78. data/vendor/spidermonkey/fdlibm/e_asin.c +156 -0
  79. data/vendor/spidermonkey/fdlibm/e_atan2.c +165 -0
  80. data/vendor/spidermonkey/fdlibm/e_atanh.c +110 -0
  81. data/vendor/spidermonkey/fdlibm/e_cosh.c +133 -0
  82. data/vendor/spidermonkey/fdlibm/e_exp.c +202 -0
  83. data/vendor/spidermonkey/fdlibm/e_fmod.c +184 -0
  84. data/vendor/spidermonkey/fdlibm/e_gamma.c +71 -0
  85. data/vendor/spidermonkey/fdlibm/e_gamma_r.c +70 -0
  86. data/vendor/spidermonkey/fdlibm/e_hypot.c +173 -0
  87. data/vendor/spidermonkey/fdlibm/e_j0.c +524 -0
  88. data/vendor/spidermonkey/fdlibm/e_j1.c +523 -0
  89. data/vendor/spidermonkey/fdlibm/e_jn.c +315 -0
  90. data/vendor/spidermonkey/fdlibm/e_lgamma.c +71 -0
  91. data/vendor/spidermonkey/fdlibm/e_lgamma_r.c +347 -0
  92. data/vendor/spidermonkey/fdlibm/e_log.c +184 -0
  93. data/vendor/spidermonkey/fdlibm/e_log10.c +134 -0
  94. data/vendor/spidermonkey/fdlibm/e_pow.c +386 -0
  95. data/vendor/spidermonkey/fdlibm/e_rem_pio2.c +222 -0
  96. data/vendor/spidermonkey/fdlibm/e_remainder.c +120 -0
  97. data/vendor/spidermonkey/fdlibm/e_scalb.c +89 -0
  98. data/vendor/spidermonkey/fdlibm/e_sinh.c +122 -0
  99. data/vendor/spidermonkey/fdlibm/e_sqrt.c +497 -0
  100. data/vendor/spidermonkey/fdlibm/fdlibm.h +273 -0
  101. data/vendor/spidermonkey/fdlibm/fdlibm.mak +1453 -0
  102. data/vendor/spidermonkey/fdlibm/fdlibm.mdp +0 -0
  103. data/vendor/spidermonkey/fdlibm/k_cos.c +135 -0
  104. data/vendor/spidermonkey/fdlibm/k_rem_pio2.c +354 -0
  105. data/vendor/spidermonkey/fdlibm/k_sin.c +114 -0
  106. data/vendor/spidermonkey/fdlibm/k_standard.c +785 -0
  107. data/vendor/spidermonkey/fdlibm/k_tan.c +170 -0
  108. data/vendor/spidermonkey/fdlibm/s_asinh.c +101 -0
  109. data/vendor/spidermonkey/fdlibm/s_atan.c +175 -0
  110. data/vendor/spidermonkey/fdlibm/s_cbrt.c +133 -0
  111. data/vendor/spidermonkey/fdlibm/s_ceil.c +120 -0
  112. data/vendor/spidermonkey/fdlibm/s_copysign.c +72 -0
  113. data/vendor/spidermonkey/fdlibm/s_cos.c +118 -0
  114. data/vendor/spidermonkey/fdlibm/s_erf.c +356 -0
  115. data/vendor/spidermonkey/fdlibm/s_expm1.c +267 -0
  116. data/vendor/spidermonkey/fdlibm/s_fabs.c +70 -0
  117. data/vendor/spidermonkey/fdlibm/s_finite.c +71 -0
  118. data/vendor/spidermonkey/fdlibm/s_floor.c +121 -0
  119. data/vendor/spidermonkey/fdlibm/s_frexp.c +99 -0
  120. data/vendor/spidermonkey/fdlibm/s_ilogb.c +85 -0
  121. data/vendor/spidermonkey/fdlibm/s_isnan.c +74 -0
  122. data/vendor/spidermonkey/fdlibm/s_ldexp.c +66 -0
  123. data/vendor/spidermonkey/fdlibm/s_lib_version.c +73 -0
  124. data/vendor/spidermonkey/fdlibm/s_log1p.c +211 -0
  125. data/vendor/spidermonkey/fdlibm/s_logb.c +79 -0
  126. data/vendor/spidermonkey/fdlibm/s_matherr.c +64 -0
  127. data/vendor/spidermonkey/fdlibm/s_modf.c +132 -0
  128. data/vendor/spidermonkey/fdlibm/s_nextafter.c +124 -0
  129. data/vendor/spidermonkey/fdlibm/s_rint.c +131 -0
  130. data/vendor/spidermonkey/fdlibm/s_scalbn.c +107 -0
  131. data/vendor/spidermonkey/fdlibm/s_signgam.c +40 -0
  132. data/vendor/spidermonkey/fdlibm/s_significand.c +68 -0
  133. data/vendor/spidermonkey/fdlibm/s_sin.c +118 -0
  134. data/vendor/spidermonkey/fdlibm/s_tan.c +112 -0
  135. data/vendor/spidermonkey/fdlibm/s_tanh.c +122 -0
  136. data/vendor/spidermonkey/fdlibm/w_acos.c +78 -0
  137. data/vendor/spidermonkey/fdlibm/w_acosh.c +78 -0
  138. data/vendor/spidermonkey/fdlibm/w_asin.c +80 -0
  139. data/vendor/spidermonkey/fdlibm/w_atan2.c +79 -0
  140. data/vendor/spidermonkey/fdlibm/w_atanh.c +81 -0
  141. data/vendor/spidermonkey/fdlibm/w_cosh.c +77 -0
  142. data/vendor/spidermonkey/fdlibm/w_exp.c +88 -0
  143. data/vendor/spidermonkey/fdlibm/w_fmod.c +78 -0
  144. data/vendor/spidermonkey/fdlibm/w_gamma.c +85 -0
  145. data/vendor/spidermonkey/fdlibm/w_gamma_r.c +81 -0
  146. data/vendor/spidermonkey/fdlibm/w_hypot.c +78 -0
  147. data/vendor/spidermonkey/fdlibm/w_j0.c +105 -0
  148. data/vendor/spidermonkey/fdlibm/w_j1.c +106 -0
  149. data/vendor/spidermonkey/fdlibm/w_jn.c +128 -0
  150. data/vendor/spidermonkey/fdlibm/w_lgamma.c +85 -0
  151. data/vendor/spidermonkey/fdlibm/w_lgamma_r.c +81 -0
  152. data/vendor/spidermonkey/fdlibm/w_log.c +78 -0
  153. data/vendor/spidermonkey/fdlibm/w_log10.c +81 -0
  154. data/vendor/spidermonkey/fdlibm/w_pow.c +99 -0
  155. data/vendor/spidermonkey/fdlibm/w_remainder.c +77 -0
  156. data/vendor/spidermonkey/fdlibm/w_scalb.c +95 -0
  157. data/vendor/spidermonkey/fdlibm/w_sinh.c +77 -0
  158. data/vendor/spidermonkey/fdlibm/w_sqrt.c +77 -0
  159. data/vendor/spidermonkey/javascript-trace.d +73 -0
  160. data/vendor/spidermonkey/js.c +3951 -0
  161. data/vendor/spidermonkey/js.mak +4438 -0
  162. data/vendor/spidermonkey/js.mdp +0 -0
  163. data/vendor/spidermonkey/js.msg +307 -0
  164. data/vendor/spidermonkey/js.pkg +2 -0
  165. data/vendor/spidermonkey/js3240.rc +79 -0
  166. data/vendor/spidermonkey/jsOS240.def +654 -0
  167. data/vendor/spidermonkey/jsapi.c +5836 -0
  168. data/vendor/spidermonkey/jsapi.h +2624 -0
  169. data/vendor/spidermonkey/jsarena.c +450 -0
  170. data/vendor/spidermonkey/jsarena.h +318 -0
  171. data/vendor/spidermonkey/jsarray.c +2988 -0
  172. data/vendor/spidermonkey/jsarray.h +124 -0
  173. data/vendor/spidermonkey/jsatom.c +1045 -0
  174. data/vendor/spidermonkey/jsatom.h +442 -0
  175. data/vendor/spidermonkey/jsbit.h +253 -0
  176. data/vendor/spidermonkey/jsbool.c +176 -0
  177. data/vendor/spidermonkey/jsbool.h +73 -0
  178. data/vendor/spidermonkey/jsclist.h +139 -0
  179. data/vendor/spidermonkey/jscntxt.c +1348 -0
  180. data/vendor/spidermonkey/jscntxt.h +1120 -0
  181. data/vendor/spidermonkey/jscompat.h +57 -0
  182. data/vendor/spidermonkey/jsconfig.h +248 -0
  183. data/vendor/spidermonkey/jsconfig.mk +181 -0
  184. data/vendor/spidermonkey/jscpucfg.c +383 -0
  185. data/vendor/spidermonkey/jscpucfg.h +212 -0
  186. data/vendor/spidermonkey/jsdate.c +2398 -0
  187. data/vendor/spidermonkey/jsdate.h +124 -0
  188. data/vendor/spidermonkey/jsdbgapi.c +1799 -0
  189. data/vendor/spidermonkey/jsdbgapi.h +464 -0
  190. data/vendor/spidermonkey/jsdhash.c +868 -0
  191. data/vendor/spidermonkey/jsdhash.h +592 -0
  192. data/vendor/spidermonkey/jsdtoa.c +3167 -0
  193. data/vendor/spidermonkey/jsdtoa.h +130 -0
  194. data/vendor/spidermonkey/jsdtracef.c +317 -0
  195. data/vendor/spidermonkey/jsdtracef.h +77 -0
  196. data/vendor/spidermonkey/jsemit.c +6909 -0
  197. data/vendor/spidermonkey/jsemit.h +741 -0
  198. data/vendor/spidermonkey/jsexn.c +1371 -0
  199. data/vendor/spidermonkey/jsexn.h +96 -0
  200. data/vendor/spidermonkey/jsfile.c +2736 -0
  201. data/vendor/spidermonkey/jsfile.h +56 -0
  202. data/vendor/spidermonkey/jsfile.msg +90 -0
  203. data/vendor/spidermonkey/jsfun.c +2634 -0
  204. data/vendor/spidermonkey/jsfun.h +254 -0
  205. data/vendor/spidermonkey/jsgc.c +3554 -0
  206. data/vendor/spidermonkey/jsgc.h +403 -0
  207. data/vendor/spidermonkey/jshash.c +476 -0
  208. data/vendor/spidermonkey/jshash.h +151 -0
  209. data/vendor/spidermonkey/jsify.pl +485 -0
  210. data/vendor/spidermonkey/jsinterp.c +6981 -0
  211. data/vendor/spidermonkey/jsinterp.h +521 -0
  212. data/vendor/spidermonkey/jsinvoke.c +43 -0
  213. data/vendor/spidermonkey/jsiter.c +1067 -0
  214. data/vendor/spidermonkey/jsiter.h +122 -0
  215. data/vendor/spidermonkey/jskeyword.tbl +124 -0
  216. data/vendor/spidermonkey/jskwgen.c +460 -0
  217. data/vendor/spidermonkey/jslibmath.h +266 -0
  218. data/vendor/spidermonkey/jslock.c +1309 -0
  219. data/vendor/spidermonkey/jslock.h +313 -0
  220. data/vendor/spidermonkey/jslocko.asm +60 -0
  221. data/vendor/spidermonkey/jslog2.c +94 -0
  222. data/vendor/spidermonkey/jslong.c +264 -0
  223. data/vendor/spidermonkey/jslong.h +412 -0
  224. data/vendor/spidermonkey/jsmath.c +568 -0
  225. data/vendor/spidermonkey/jsmath.h +57 -0
  226. data/vendor/spidermonkey/jsnum.c +1228 -0
  227. data/vendor/spidermonkey/jsnum.h +283 -0
  228. data/vendor/spidermonkey/jsobj.c +5266 -0
  229. data/vendor/spidermonkey/jsobj.h +709 -0
  230. data/vendor/spidermonkey/jsopcode.c +5245 -0
  231. data/vendor/spidermonkey/jsopcode.h +394 -0
  232. data/vendor/spidermonkey/jsopcode.tbl +523 -0
  233. data/vendor/spidermonkey/jsotypes.h +202 -0
  234. data/vendor/spidermonkey/jsparse.c +6680 -0
  235. data/vendor/spidermonkey/jsparse.h +511 -0
  236. data/vendor/spidermonkey/jsprf.c +1262 -0
  237. data/vendor/spidermonkey/jsprf.h +150 -0
  238. data/vendor/spidermonkey/jsproto.tbl +128 -0
  239. data/vendor/spidermonkey/jsprvtd.h +267 -0
  240. data/vendor/spidermonkey/jspubtd.h +744 -0
  241. data/vendor/spidermonkey/jsregexp.c +4352 -0
  242. data/vendor/spidermonkey/jsregexp.h +183 -0
  243. data/vendor/spidermonkey/jsreops.tbl +145 -0
  244. data/vendor/spidermonkey/jsscan.c +2003 -0
  245. data/vendor/spidermonkey/jsscan.h +387 -0
  246. data/vendor/spidermonkey/jsscope.c +1948 -0
  247. data/vendor/spidermonkey/jsscope.h +418 -0
  248. data/vendor/spidermonkey/jsscript.c +1832 -0
  249. data/vendor/spidermonkey/jsscript.h +287 -0
  250. data/vendor/spidermonkey/jsshell.msg +50 -0
  251. data/vendor/spidermonkey/jsstddef.h +83 -0
  252. data/vendor/spidermonkey/jsstr.c +5004 -0
  253. data/vendor/spidermonkey/jsstr.h +641 -0
  254. data/vendor/spidermonkey/jstypes.h +475 -0
  255. data/vendor/spidermonkey/jsutil.c +345 -0
  256. data/vendor/spidermonkey/jsutil.h +157 -0
  257. data/vendor/spidermonkey/jsxdrapi.c +800 -0
  258. data/vendor/spidermonkey/jsxdrapi.h +218 -0
  259. data/vendor/spidermonkey/jsxml.c +8471 -0
  260. data/vendor/spidermonkey/jsxml.h +349 -0
  261. data/vendor/spidermonkey/lock_SunOS.s +119 -0
  262. data/vendor/spidermonkey/perfect.js +39 -0
  263. data/vendor/spidermonkey/plify_jsdhash.sed +36 -0
  264. data/vendor/spidermonkey/prmjtime.c +846 -0
  265. data/vendor/spidermonkey/prmjtime.h +103 -0
  266. data/vendor/spidermonkey/resource.h +15 -0
  267. data/vendor/spidermonkey/rules.mk +197 -0
  268. data/vendor/spidermonkey/win32.order +384 -0
  269. metadata +4 -3
@@ -0,0 +1,96 @@
1
+ /* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
2
+ *
3
+ * ***** BEGIN LICENSE BLOCK *****
4
+ * Version: MPL 1.1/GPL 2.0/LGPL 2.1
5
+ *
6
+ * The contents of this file are subject to the Mozilla Public License Version
7
+ * 1.1 (the "License"); you may not use this file except in compliance with
8
+ * the License. You may obtain a copy of the License at
9
+ * http://www.mozilla.org/MPL/
10
+ *
11
+ * Software distributed under the License is distributed on an "AS IS" basis,
12
+ * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
13
+ * for the specific language governing rights and limitations under the
14
+ * License.
15
+ *
16
+ * The Original Code is Mozilla Communicator client code, released
17
+ * March 31, 1998.
18
+ *
19
+ * The Initial Developer of the Original Code is
20
+ * Netscape Communications Corporation.
21
+ * Portions created by the Initial Developer are Copyright (C) 1998
22
+ * the Initial Developer. All Rights Reserved.
23
+ *
24
+ * Contributor(s):
25
+ *
26
+ * Alternatively, the contents of this file may be used under the terms of
27
+ * either of the GNU General Public License Version 2 or later (the "GPL"),
28
+ * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
29
+ * in which case the provisions of the GPL or the LGPL are applicable instead
30
+ * of those above. If you wish to allow use of your version of this file only
31
+ * under the terms of either the GPL or the LGPL, and not to allow others to
32
+ * use your version of this file under the terms of the MPL, indicate your
33
+ * decision by deleting the provisions above and replace them with the notice
34
+ * and other provisions required by the GPL or the LGPL. If you do not delete
35
+ * the provisions above, a recipient may use your version of this file under
36
+ * the terms of any one of the MPL, the GPL or the LGPL.
37
+ *
38
+ * ***** END LICENSE BLOCK ***** */
39
+
40
+ /*
41
+ * JS runtime exception classes.
42
+ */
43
+
44
+ #ifndef jsexn_h___
45
+ #define jsexn_h___
46
+
47
+ JS_BEGIN_EXTERN_C
48
+
49
+ extern JSClass js_ErrorClass;
50
+
51
+ /*
52
+ * Initialize the exception constructor/prototype hierarchy.
53
+ */
54
+ extern JSObject *
55
+ js_InitExceptionClasses(JSContext *cx, JSObject *obj);
56
+
57
+ /*
58
+ * Given a JSErrorReport, check to see if there is an exception associated with
59
+ * the error number. If there is, then create an appropriate exception object,
60
+ * set it as the pending exception, and set the JSREPORT_EXCEPTION flag on the
61
+ * error report. Exception-aware host error reporters should probably ignore
62
+ * error reports so flagged. Returns JS_TRUE if an associated exception is
63
+ * found and set, JS_FALSE otherwise..
64
+ */
65
+ extern JSBool
66
+ js_ErrorToException(JSContext *cx, const char *message, JSErrorReport *reportp);
67
+
68
+ /*
69
+ * Called if a JS API call to js_Execute or js_InternalCall fails; calls the
70
+ * error reporter with the error report associated with any uncaught exception
71
+ * that has been raised. Returns true if there was an exception pending, and
72
+ * the error reporter was actually called.
73
+ *
74
+ * The JSErrorReport * that the error reporter is called with is currently
75
+ * associated with a JavaScript object, and is not guaranteed to persist after
76
+ * the object is collected. Any persistent uses of the JSErrorReport contents
77
+ * should make their own copy.
78
+ *
79
+ * The flags field of the JSErrorReport will have the JSREPORT_EXCEPTION flag
80
+ * set; embeddings that want to silently propagate JavaScript exceptions to
81
+ * other contexts may want to use an error reporter that ignores errors with
82
+ * this flag.
83
+ */
84
+ extern JSBool
85
+ js_ReportUncaughtException(JSContext *cx);
86
+
87
+ extern JSErrorReport *
88
+ js_ErrorFromException(JSContext *cx, jsval exn);
89
+
90
+ extern const JSErrorFormatString *
91
+ js_GetLocalizedErrorMessage(JSContext* cx, void *userRef, const char *locale,
92
+ const uintN errorNumber);
93
+
94
+ JS_END_EXTERN_C
95
+
96
+ #endif /* jsexn_h___ */
@@ -0,0 +1,2736 @@
1
+ /* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
2
+ * vim: set ts=8 sw=4 et tw=78:
3
+ *
4
+ * ***** BEGIN LICENSE BLOCK *****
5
+ * Version: MPL 1.1/GPL 2.0/LGPL 2.1
6
+ *
7
+ * The contents of this file are subject to the Mozilla Public License Version
8
+ * 1.1 (the "License"); you may not use this file except in compliance with
9
+ * the License. You may obtain a copy of the License at
10
+ * http://www.mozilla.org/MPL/
11
+ *
12
+ * Software distributed under the License is distributed on an "AS IS" basis,
13
+ * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
14
+ * for the specific language governing rights and limitations under the
15
+ * License.
16
+ *
17
+ * The Original Code is Mozilla Communicator client code, released
18
+ * March 31, 1998.
19
+ *
20
+ * The Initial Developer of the Original Code is
21
+ * Netscape Communications Corporation.
22
+ * Portions created by the Initial Developer are Copyright (C) 1998
23
+ * the Initial Developer. All Rights Reserved.
24
+ *
25
+ * Contributor(s):
26
+ *
27
+ * Alternatively, the contents of this file may be used under the terms of
28
+ * either of the GNU General Public License Version 2 or later (the "GPL"),
29
+ * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
30
+ * in which case the provisions of the GPL or the LGPL are applicable instead
31
+ * of those above. If you wish to allow use of your version of this file only
32
+ * under the terms of either the GPL or the LGPL, and not to allow others to
33
+ * use your version of this file under the terms of the MPL, indicate your
34
+ * decision by deleting the provisions above and replace them with the notice
35
+ * and other provisions required by the GPL or the LGPL. If you do not delete
36
+ * the provisions above, a recipient may use your version of this file under
37
+ * the terms of any one of the MPL, the GPL or the LGPL.
38
+ *
39
+ * ***** END LICENSE BLOCK ***** */
40
+
41
+ /*
42
+ * JS File object
43
+ */
44
+ #if JS_HAS_FILE_OBJECT
45
+
46
+ #include "jsstddef.h"
47
+ #include "jsfile.h"
48
+
49
+ /* ----------------- Platform-specific includes and defines ----------------- */
50
+ #if defined(XP_WIN) || defined(XP_OS2)
51
+ # include <direct.h>
52
+ # include <io.h>
53
+ # include <sys/types.h>
54
+ # include <sys/stat.h>
55
+ # define FILESEPARATOR '\\'
56
+ # define FILESEPARATOR2 '/'
57
+ # define CURRENT_DIR "c:\\"
58
+ # define POPEN _popen
59
+ # define PCLOSE _pclose
60
+ #elif defined(XP_UNIX) || defined(XP_BEOS)
61
+ # include <strings.h>
62
+ # include <stdio.h>
63
+ # include <stdlib.h>
64
+ # include <unistd.h>
65
+ # define FILESEPARATOR '/'
66
+ # define FILESEPARATOR2 '\0'
67
+ # define CURRENT_DIR "/"
68
+ # define POPEN popen
69
+ # define PCLOSE pclose
70
+ #endif
71
+
72
+ /* --------------- Platform-independent includes and defines ---------------- */
73
+ #include "jsapi.h"
74
+ #include "jsatom.h"
75
+ #include "jscntxt.h"
76
+ #include "jsdate.h"
77
+ #include "jsdbgapi.h"
78
+ #include "jsemit.h"
79
+ #include "jsfun.h"
80
+ #include "jslock.h"
81
+ #include "jsobj.h"
82
+ #include "jsparse.h"
83
+ #include "jsscan.h"
84
+ #include "jsscope.h"
85
+ #include "jsscript.h"
86
+ #include "jsstr.h"
87
+ #include "jsutil.h" /* Added by JSIFY */
88
+ #include <string.h>
89
+
90
+ /* NSPR dependencies */
91
+ #include "prio.h"
92
+ #include "prerror.h"
93
+
94
+ #define SPECIAL_FILE_STRING "Special File"
95
+ #define CURRENTDIR_PROPERTY "currentDir"
96
+ #define SEPARATOR_PROPERTY "separator"
97
+ #define FILE_CONSTRUCTOR "File"
98
+ #define PIPE_SYMBOL '|'
99
+
100
+ #define ASCII 0
101
+ #define UTF8 1
102
+ #define UCS2 2
103
+
104
+ #define asciistring "text"
105
+ #define utfstring "binary"
106
+ #define unicodestring "unicode"
107
+
108
+ #define MAX_PATH_LENGTH 1024
109
+ #define MODE_SIZE 256
110
+ #define NUMBER_SIZE 32
111
+ #define MAX_LINE_LENGTH 256
112
+ #define URL_PREFIX "file://"
113
+
114
+ #define STDINPUT_NAME "Standard input stream"
115
+ #define STDOUTPUT_NAME "Standard output stream"
116
+ #define STDERROR_NAME "Standard error stream"
117
+
118
+ #define RESOLVE_PATH js_canonicalPath /* js_absolutePath */
119
+
120
+ /* Error handling */
121
+ typedef enum JSFileErrNum {
122
+ #define MSG_DEF(name, number, count, exception, format) \
123
+ name = number,
124
+ #include "jsfile.msg"
125
+ #undef MSG_DEF
126
+ JSFileErr_Limit
127
+ #undef MSGDEF
128
+ } JSFileErrNum;
129
+
130
+ #define JSFILE_HAS_DFLT_MSG_STRINGS 1
131
+
132
+ JSErrorFormatString JSFile_ErrorFormatString[JSFileErr_Limit] = {
133
+ #if JSFILE_HAS_DFLT_MSG_STRINGS
134
+ #define MSG_DEF(name, number, count, exception, format) \
135
+ { format, count },
136
+ #else
137
+ #define MSG_DEF(name, number, count, exception, format) \
138
+ { NULL, count },
139
+ #endif
140
+ #include "jsfile.msg"
141
+ #undef MSG_DEF
142
+ };
143
+
144
+ const JSErrorFormatString *
145
+ JSFile_GetErrorMessage(void *userRef, const char *locale,
146
+ const uintN errorNumber)
147
+ {
148
+ if ((errorNumber > 0) && (errorNumber < JSFileErr_Limit))
149
+ return &JSFile_ErrorFormatString[errorNumber];
150
+ else
151
+ return NULL;
152
+ }
153
+
154
+ #define JSFILE_CHECK_NATIVE(op) \
155
+ if (file->isNative) { \
156
+ JS_ReportWarning(cx, "Cannot call or access \"%s\" on native file %s",\
157
+ op, file->path); \
158
+ goto out; \
159
+ }
160
+
161
+ #define JSFILE_CHECK_WRITE \
162
+ if (!file->isOpen) { \
163
+ JS_ReportWarning(cx, \
164
+ "File %s is closed, will open it for writing, proceeding", \
165
+ file->path); \
166
+ js_FileOpen(cx, obj, file, "write,append,create"); \
167
+ } \
168
+ if (!js_canWrite(cx, file)) { \
169
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL, \
170
+ JSFILEMSG_CANNOT_WRITE, file->path); \
171
+ goto out; \
172
+ }
173
+
174
+ #define JSFILE_CHECK_READ \
175
+ if (!file->isOpen) { \
176
+ JS_ReportWarning(cx, \
177
+ "File %s is closed, will open it for reading, proceeding", \
178
+ file->path); \
179
+ js_FileOpen(cx, obj, file, "read"); \
180
+ } \
181
+ if (!js_canRead(cx, file)) { \
182
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL, \
183
+ JSFILEMSG_CANNOT_READ, file->path); \
184
+ goto out; \
185
+ }
186
+
187
+ #define JSFILE_CHECK_OPEN(op) \
188
+ if (!file->isOpen) { \
189
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL, \
190
+ JSFILEMSG_FILE_MUST_BE_OPEN, op); \
191
+ goto out; \
192
+ }
193
+
194
+ #define JSFILE_CHECK_CLOSED(op) \
195
+ if (file->isOpen) { \
196
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL, \
197
+ JSFILEMSG_FILE_MUST_BE_CLOSED, op); \
198
+ goto out; \
199
+ }
200
+
201
+ #define JSFILE_CHECK_ONE_ARG(op) \
202
+ if (argc != 1) { \
203
+ char str[NUMBER_SIZE]; \
204
+ sprintf(str, "%d", argc); \
205
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL, \
206
+ JSFILEMSG_EXPECTS_ONE_ARG_ERROR, op, str); \
207
+ goto out; \
208
+ }
209
+
210
+
211
+ /*
212
+ Security mechanism, should define a callback for this.
213
+ The parameters are as follows:
214
+ SECURITY_CHECK(JSContext *cx, JSPrincipals *ps, char *op_name, JSFile *file)
215
+ XXX Should this be a real function returning a JSBool result (and getting
216
+ some typesafety help from the compiler?).
217
+ */
218
+ #define SECURITY_CHECK(cx, ps, op, file) \
219
+ /* Define a callback here... */
220
+
221
+
222
+ /* Structure representing the file internally */
223
+ typedef struct JSFile {
224
+ char *path; /* the path to the file. */
225
+ JSBool isOpen;
226
+ int32 mode; /* mode used to open the file: read, write, append, create, etc.. */
227
+ int32 type; /* Asciiz, utf, unicode */
228
+ char byteBuffer[3]; /* bytes read in advance by js_FileRead ( UTF8 encoding ) */
229
+ jsint nbBytesInBuf; /* number of bytes stored in the buffer above */
230
+ jschar charBuffer; /* character read in advance by readln ( mac files only ) */
231
+ JSBool charBufferUsed; /* flag indicating if the buffer above is being used */
232
+ JSBool hasRandomAccess;/* can the file be randomly accessed? false for stdin, and
233
+ UTF-encoded files. */
234
+ JSBool hasAutoflush; /* should we force a flush for each line break? */
235
+ JSBool isNative; /* if the file is using OS-specific file FILE type */
236
+ /* We can actually put the following two in a union since they should never be used at the same time */
237
+ PRFileDesc *handle; /* the handle for the file, if open. */
238
+ FILE *nativehandle; /* native handle, for stuff NSPR doesn't do. */
239
+ JSBool isPipe; /* if the file is really an OS pipe */
240
+ } JSFile;
241
+
242
+ /* a few forward declarations... */
243
+ JS_PUBLIC_API(JSObject*) js_NewFileObject(JSContext *cx, char *filename);
244
+ static JSBool file_open(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval);
245
+ static JSBool file_close(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval);
246
+
247
+ /* New filename manipulation procesures */
248
+ /* assumes we don't have leading/trailing spaces */
249
+ static JSBool
250
+ js_filenameHasAPipe(const char *filename)
251
+ {
252
+ if (!filename)
253
+ return JS_FALSE;
254
+
255
+ return filename[0] == PIPE_SYMBOL ||
256
+ filename[strlen(filename) - 1] == PIPE_SYMBOL;
257
+ }
258
+
259
+ static JSBool
260
+ js_isAbsolute(const char *name)
261
+ {
262
+ #if defined(XP_WIN) || defined(XP_OS2)
263
+ return *name && name[1] == ':';
264
+ #else
265
+ return (name[0]
266
+ # if defined(XP_UNIX) || defined(XP_BEOS)
267
+ ==
268
+ # else
269
+ !=
270
+ # endif
271
+ FILESEPARATOR);
272
+ #endif
273
+ }
274
+
275
+ /*
276
+ * Concatenates base and name to produce a valid filename.
277
+ * Returned string must be freed.
278
+ */
279
+ static char*
280
+ js_combinePath(JSContext *cx, const char *base, const char *name)
281
+ {
282
+ int len = strlen(base);
283
+ char* result = JS_malloc(cx, len + strlen(name) + 2);
284
+
285
+ if (!result)
286
+ return NULL;
287
+
288
+ strcpy(result, base);
289
+
290
+ if (base[len - 1] != FILESEPARATOR && base[len - 1] != FILESEPARATOR2) {
291
+ result[len] = FILESEPARATOR;
292
+ result[len + 1] = '\0';
293
+ }
294
+ strcat(result, name);
295
+ return result;
296
+ }
297
+
298
+ /* Extract the last component from a path name. Returned string must be freed */
299
+ static char *
300
+ js_fileBaseName(JSContext *cx, const char *pathname)
301
+ {
302
+ jsint index, aux;
303
+ char *result;
304
+
305
+ index = strlen(pathname)-1;
306
+
307
+ /* Chop off trailing seperators. */
308
+ while (index > 0 && (pathname[index]==FILESEPARATOR ||
309
+ pathname[index]==FILESEPARATOR2)) {
310
+ --index;
311
+ }
312
+
313
+ aux = index;
314
+
315
+ /* Now find the next separator. */
316
+ while (index >= 0 && pathname[index] != FILESEPARATOR &&
317
+ pathname[index] != FILESEPARATOR2) {
318
+ --index;
319
+ }
320
+
321
+ /* Allocate and copy. */
322
+ result = JS_malloc(cx, aux - index + 1);
323
+ if (!result)
324
+ return NULL;
325
+ strncpy(result, pathname + index + 1, aux - index);
326
+ result[aux - index] = '\0';
327
+ return result;
328
+ }
329
+
330
+ /*
331
+ * Returns everything but the last component from a path name.
332
+ * Returned string must be freed.
333
+ */
334
+ static char *
335
+ js_fileDirectoryName(JSContext *cx, const char *pathname)
336
+ {
337
+ char *result;
338
+ const char *cp, *end;
339
+ size_t pathsize;
340
+
341
+ end = pathname + strlen(pathname);
342
+ cp = end - 1;
343
+
344
+ /* If this is already a directory, chop off the trailing /s. */
345
+ while (cp >= pathname) {
346
+ if (*cp != FILESEPARATOR && *cp != FILESEPARATOR2)
347
+ break;
348
+ --cp;
349
+ }
350
+
351
+ if (cp < pathname && end != pathname) {
352
+ /* There were just /s, return the root. */
353
+ result = JS_malloc(cx, 1 + 1); /* The separator + trailing NUL. */
354
+ result[0] = FILESEPARATOR;
355
+ result[1] = '\0';
356
+ return result;
357
+ }
358
+
359
+ /* Now chop off the last portion. */
360
+ while (cp >= pathname) {
361
+ if (*cp == FILESEPARATOR || *cp == FILESEPARATOR2)
362
+ break;
363
+ --cp;
364
+ }
365
+
366
+ /* Check if this is a leaf. */
367
+ if (cp < pathname) {
368
+ /* It is, return "pathname/". */
369
+ if (end[-1] == FILESEPARATOR || end[-1] == FILESEPARATOR2) {
370
+ /* Already has its terminating /. */
371
+ return JS_strdup(cx, pathname);
372
+ }
373
+
374
+ pathsize = end - pathname + 1;
375
+ result = JS_malloc(cx, pathsize + 1);
376
+ if (!result)
377
+ return NULL;
378
+
379
+ strcpy(result, pathname);
380
+ result[pathsize - 1] = FILESEPARATOR;
381
+ result[pathsize] = '\0';
382
+
383
+ return result;
384
+ }
385
+
386
+ /* Return everything up to and including the seperator. */
387
+ pathsize = cp - pathname + 1;
388
+ result = JS_malloc(cx, pathsize + 1);
389
+ if (!result)
390
+ return NULL;
391
+
392
+ strncpy(result, pathname, pathsize);
393
+ result[pathsize] = '\0';
394
+
395
+ return result;
396
+ }
397
+
398
+ static char *
399
+ js_absolutePath(JSContext *cx, const char * path)
400
+ {
401
+ JSObject *obj;
402
+ JSString *str;
403
+ jsval prop;
404
+
405
+ if (js_isAbsolute(path)) {
406
+ return JS_strdup(cx, path);
407
+ } else {
408
+ obj = JS_GetGlobalObject(cx);
409
+ if (!JS_GetProperty(cx, obj, FILE_CONSTRUCTOR, &prop)) {
410
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
411
+ JSFILEMSG_FILE_CONSTRUCTOR_UNDEFINED_ERROR);
412
+ return JS_strdup(cx, path);
413
+ }
414
+
415
+ obj = JSVAL_TO_OBJECT(prop);
416
+ if (!JS_GetProperty(cx, obj, CURRENTDIR_PROPERTY, &prop)) {
417
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
418
+ JSFILEMSG_FILE_CURRENTDIR_UNDEFINED_ERROR);
419
+ return JS_strdup(cx, path);
420
+ }
421
+
422
+ str = JS_ValueToString(cx, prop);
423
+ if (!str)
424
+ return JS_strdup(cx, path);
425
+
426
+ /* should we have an array of curr dirs indexed by drive for windows? */
427
+ return js_combinePath(cx, JS_GetStringBytes(str), path);
428
+ }
429
+ }
430
+
431
+ /* Side effect: will remove spaces in the beginning/end of the filename */
432
+ static char *
433
+ js_canonicalPath(JSContext *cx, char *oldpath)
434
+ {
435
+ char *tmp;
436
+ char *path = oldpath;
437
+ char *base, *dir, *current, *result;
438
+ jsint c;
439
+ jsint back = 0;
440
+ unsigned int i = 0, j = strlen(path)-1;
441
+
442
+ /* This is probably optional */
443
+ /* Remove possible spaces in the beginning and end */
444
+ while (i < j && path[i] == ' ')
445
+ i++;
446
+ while (j >= 0 && path[j] == ' ')
447
+ j--;
448
+
449
+ tmp = JS_malloc(cx, j-i+2);
450
+ if (!tmp)
451
+ return NULL;
452
+
453
+ strncpy(tmp, path + i, j - i + 1);
454
+ tmp[j - i + 1] = '\0';
455
+
456
+ path = tmp;
457
+
458
+ /* Pipe support. */
459
+ if (js_filenameHasAPipe(path))
460
+ return path;
461
+
462
+ /* file:// support. */
463
+ if (!strncmp(path, URL_PREFIX, strlen(URL_PREFIX))) {
464
+ tmp = js_canonicalPath(cx, path + strlen(URL_PREFIX));
465
+ JS_free(cx, path);
466
+ return tmp;
467
+ }
468
+
469
+ if (!js_isAbsolute(path)) {
470
+ tmp = js_absolutePath(cx, path);
471
+ if (!tmp)
472
+ return NULL;
473
+ JS_free(cx, path);
474
+ path = tmp;
475
+ }
476
+
477
+ result = JS_strdup(cx, "");
478
+
479
+ current = path;
480
+
481
+ base = js_fileBaseName(cx, current);
482
+ dir = js_fileDirectoryName(cx, current);
483
+
484
+ while (strcmp(dir, current)) {
485
+ if (!strcmp(base, "..")) {
486
+ back++;
487
+ } else {
488
+ if (back > 0) {
489
+ back--;
490
+ } else {
491
+ tmp = result;
492
+ result = JS_malloc(cx, strlen(base) + 1 + strlen(tmp) + 1);
493
+ if (!result)
494
+ goto out;
495
+
496
+ strcpy(result, base);
497
+ c = strlen(result);
498
+ if (*tmp) {
499
+ result[c] = FILESEPARATOR;
500
+ result[c + 1] = '\0';
501
+ strcat(result, tmp);
502
+ }
503
+ JS_free(cx, tmp);
504
+ }
505
+ }
506
+ JS_free(cx, current);
507
+ JS_free(cx, base);
508
+ current = dir;
509
+ base = js_fileBaseName(cx, current);
510
+ dir = js_fileDirectoryName(cx, current);
511
+ }
512
+
513
+ tmp = result;
514
+ result = JS_malloc(cx, strlen(dir)+1+strlen(tmp)+1);
515
+ if (!result)
516
+ goto out;
517
+
518
+ strcpy(result, dir);
519
+ c = strlen(result);
520
+ if (tmp[0]!='\0') {
521
+ if ((result[c-1]!=FILESEPARATOR)&&(result[c-1]!=FILESEPARATOR2)) {
522
+ result[c] = FILESEPARATOR;
523
+ result[c+1] = '\0';
524
+ }
525
+ strcat(result, tmp);
526
+ }
527
+
528
+ out:
529
+ if (tmp)
530
+ JS_free(cx, tmp);
531
+ if (dir)
532
+ JS_free(cx, dir);
533
+ if (base)
534
+ JS_free(cx, base);
535
+ if (current)
536
+ JS_free(cx, current);
537
+
538
+ return result;
539
+ }
540
+
541
+ /* -------------------------- Text conversion ------------------------------- */
542
+ /* The following is ripped from libi18n/unicvt.c and include files.. */
543
+
544
+ /*
545
+ * UTF8 defines and macros
546
+ */
547
+ #define ONE_OCTET_BASE 0x00 /* 0xxxxxxx */
548
+ #define ONE_OCTET_MASK 0x7F /* x1111111 */
549
+ #define CONTINUING_OCTET_BASE 0x80 /* 10xxxxxx */
550
+ #define CONTINUING_OCTET_MASK 0x3F /* 00111111 */
551
+ #define TWO_OCTET_BASE 0xC0 /* 110xxxxx */
552
+ #define TWO_OCTET_MASK 0x1F /* 00011111 */
553
+ #define THREE_OCTET_BASE 0xE0 /* 1110xxxx */
554
+ #define THREE_OCTET_MASK 0x0F /* 00001111 */
555
+ #define FOUR_OCTET_BASE 0xF0 /* 11110xxx */
556
+ #define FOUR_OCTET_MASK 0x07 /* 00000111 */
557
+ #define FIVE_OCTET_BASE 0xF8 /* 111110xx */
558
+ #define FIVE_OCTET_MASK 0x03 /* 00000011 */
559
+ #define SIX_OCTET_BASE 0xFC /* 1111110x */
560
+ #define SIX_OCTET_MASK 0x01 /* 00000001 */
561
+
562
+ #define IS_UTF8_1ST_OF_1(x) (( (x)&~ONE_OCTET_MASK ) == ONE_OCTET_BASE)
563
+ #define IS_UTF8_1ST_OF_2(x) (( (x)&~TWO_OCTET_MASK ) == TWO_OCTET_BASE)
564
+ #define IS_UTF8_1ST_OF_3(x) (( (x)&~THREE_OCTET_MASK) == THREE_OCTET_BASE)
565
+ #define IS_UTF8_1ST_OF_4(x) (( (x)&~FOUR_OCTET_MASK ) == FOUR_OCTET_BASE)
566
+ #define IS_UTF8_1ST_OF_5(x) (( (x)&~FIVE_OCTET_MASK ) == FIVE_OCTET_BASE)
567
+ #define IS_UTF8_1ST_OF_6(x) (( (x)&~SIX_OCTET_MASK ) == SIX_OCTET_BASE)
568
+ #define IS_UTF8_2ND_THRU_6TH(x) \
569
+ (( (x)&~CONTINUING_OCTET_MASK ) == CONTINUING_OCTET_BASE)
570
+ #define IS_UTF8_1ST_OF_UCS2(x) \
571
+ IS_UTF8_1ST_OF_1(x) \
572
+ || IS_UTF8_1ST_OF_2(x) \
573
+ || IS_UTF8_1ST_OF_3(x)
574
+
575
+
576
+ #define MAX_UCS2 0xFFFF
577
+ #define DEFAULT_CHAR 0x003F /* Default char is "?" */
578
+ #define BYTE_MASK 0xBF
579
+ #define BYTE_MARK 0x80
580
+
581
+
582
+ /* Function: one_ucs2_to_utf8_char
583
+ *
584
+ * Function takes one UCS-2 char and writes it to a UTF-8 buffer.
585
+ * We need a UTF-8 buffer because we don't know before this
586
+ * function how many bytes of utf-8 data will be written. It also
587
+ * takes a pointer to the end of the UTF-8 buffer so that we don't
588
+ * overwrite data. This function returns the number of UTF-8 bytes
589
+ * of data written, or -1 if the buffer would have been overrun.
590
+ */
591
+
592
+ #define LINE_SEPARATOR 0x2028
593
+ #define PARAGRAPH_SEPARATOR 0x2029
594
+ static int16 one_ucs2_to_utf8_char(unsigned char *tobufp,
595
+ unsigned char *tobufendp,
596
+ uint16 onechar)
597
+ {
598
+ int16 numUTF8bytes = 0;
599
+
600
+ if (onechar == LINE_SEPARATOR || onechar == PARAGRAPH_SEPARATOR) {
601
+ strcpy((char*)tobufp, "\n");
602
+ return strlen((char*)tobufp);
603
+ }
604
+
605
+ if (onechar < 0x80) {
606
+ numUTF8bytes = 1;
607
+ } else if (onechar < 0x800) {
608
+ numUTF8bytes = 2;
609
+ } else {
610
+ /* 0x800 >= onechar <= MAX_UCS2 */
611
+ numUTF8bytes = 3;
612
+ }
613
+
614
+ tobufp += numUTF8bytes;
615
+
616
+ /* return error if we don't have space for the whole character */
617
+ if (tobufp > tobufendp) {
618
+ return(-1);
619
+ }
620
+
621
+ switch(numUTF8bytes) {
622
+ case 3: *--tobufp = (onechar | BYTE_MARK) & BYTE_MASK; onechar >>=6;
623
+ *--tobufp = (onechar | BYTE_MARK) & BYTE_MASK; onechar >>=6;
624
+ *--tobufp = onechar | THREE_OCTET_BASE;
625
+ break;
626
+
627
+ case 2: *--tobufp = (onechar | BYTE_MARK) & BYTE_MASK; onechar >>=6;
628
+ *--tobufp = onechar | TWO_OCTET_BASE;
629
+ break;
630
+
631
+ case 1: *--tobufp = (unsigned char)onechar;
632
+ break;
633
+ }
634
+
635
+ return numUTF8bytes;
636
+ }
637
+
638
+ /*
639
+ * utf8_to_ucs2_char
640
+ *
641
+ * Convert a utf8 multibyte character to ucs2
642
+ *
643
+ * inputs: pointer to utf8 character(s)
644
+ * length of utf8 buffer ("read" length limit)
645
+ * pointer to return ucs2 character
646
+ *
647
+ * outputs: number of bytes in the utf8 character
648
+ * -1 if not a valid utf8 character sequence
649
+ * -2 if the buffer is too short
650
+ */
651
+ static int16
652
+ utf8_to_ucs2_char(const unsigned char *utf8p, int16 buflen, uint16 *ucs2p)
653
+ {
654
+ uint16 lead, cont1, cont2;
655
+
656
+ /*
657
+ * Check for minimum buffer length
658
+ */
659
+ if ((buflen < 1) || (utf8p == NULL)) {
660
+ return -2;
661
+ }
662
+ lead = (uint16) (*utf8p);
663
+
664
+ /*
665
+ * Check for a one octet sequence
666
+ */
667
+ if (IS_UTF8_1ST_OF_1(lead)) {
668
+ *ucs2p = lead & ONE_OCTET_MASK;
669
+ return 1;
670
+ }
671
+
672
+ /*
673
+ * Check for a two octet sequence
674
+ */
675
+ if (IS_UTF8_1ST_OF_2(*utf8p)) {
676
+ if (buflen < 2)
677
+ return -2;
678
+ cont1 = (uint16) *(utf8p+1);
679
+ if (!IS_UTF8_2ND_THRU_6TH(cont1))
680
+ return -1;
681
+ *ucs2p = (lead & TWO_OCTET_MASK) << 6;
682
+ *ucs2p |= cont1 & CONTINUING_OCTET_MASK;
683
+ return 2;
684
+ }
685
+
686
+ /*
687
+ * Check for a three octet sequence
688
+ */
689
+ else if (IS_UTF8_1ST_OF_3(lead)) {
690
+ if (buflen < 3)
691
+ return -2;
692
+ cont1 = (uint16) *(utf8p+1);
693
+ cont2 = (uint16) *(utf8p+2);
694
+ if ( (!IS_UTF8_2ND_THRU_6TH(cont1))
695
+ || (!IS_UTF8_2ND_THRU_6TH(cont2)))
696
+ return -1;
697
+ *ucs2p = (lead & THREE_OCTET_MASK) << 12;
698
+ *ucs2p |= (cont1 & CONTINUING_OCTET_MASK) << 6;
699
+ *ucs2p |= cont2 & CONTINUING_OCTET_MASK;
700
+ return 3;
701
+ }
702
+ else { /* not a valid utf8/ucs2 character */
703
+ return -1;
704
+ }
705
+ }
706
+
707
+ /* ----------------------------- Helper functions --------------------------- */
708
+ /* Ripped off from lm_win.c .. */
709
+ /* where is strcasecmp?.. for now, it's case sensitive..
710
+ *
711
+ * strcasecmp is in strings.h, but on windows it's called _stricmp...
712
+ * will need to #ifdef this
713
+ */
714
+
715
+ static int32
716
+ js_FileHasOption(JSContext *cx, const char *oldoptions, const char *name)
717
+ {
718
+ char *comma, *equal, *current;
719
+ char *options = JS_strdup(cx, oldoptions);
720
+ int32 found = 0;
721
+
722
+ current = options;
723
+ for (;;) {
724
+ comma = strchr(current, ',');
725
+ if (comma) *comma = '\0';
726
+ equal = strchr(current, '=');
727
+ if (equal) *equal = '\0';
728
+ if (strcmp(current, name) == 0) {
729
+ if (!equal || strcmp(equal + 1, "yes") == 0)
730
+ found = 1;
731
+ else
732
+ found = atoi(equal + 1);
733
+ }
734
+ if (equal) *equal = '=';
735
+ if (comma) *comma = ',';
736
+ if (found || !comma)
737
+ break;
738
+ current = comma + 1;
739
+ }
740
+ JS_free(cx, options);
741
+ return found;
742
+ }
743
+
744
+ /* empty the buffer */
745
+ static void
746
+ js_ResetBuffers(JSFile * file)
747
+ {
748
+ file->charBufferUsed = JS_FALSE;
749
+ file->nbBytesInBuf = 0;
750
+ }
751
+
752
+ /* Reset file attributes */
753
+ static void
754
+ js_ResetAttributes(JSFile * file)
755
+ {
756
+ file->mode = file->type = 0;
757
+ file->isOpen = JS_FALSE;
758
+ file->handle = NULL;
759
+ file->nativehandle = NULL;
760
+ file->hasRandomAccess = JS_TRUE; /* Innocent until proven guilty. */
761
+ file->hasAutoflush = JS_FALSE;
762
+ file->isNative = JS_FALSE;
763
+ file->isPipe = JS_FALSE;
764
+
765
+ js_ResetBuffers(file);
766
+ }
767
+
768
+ static JSBool
769
+ js_FileOpen(JSContext *cx, JSObject *obj, JSFile *file, char *mode){
770
+ JSString *type, *mask;
771
+ jsval v[2];
772
+ jsval rval;
773
+
774
+ type = JS_InternString(cx, asciistring);
775
+ mask = JS_NewStringCopyZ(cx, mode);
776
+ v[0] = STRING_TO_JSVAL(mask);
777
+ v[1] = STRING_TO_JSVAL(type);
778
+
779
+ if (!file_open(cx, obj, 2, v, &rval))
780
+ return JS_FALSE;
781
+ return JS_TRUE;
782
+ }
783
+
784
+ /* Buffered version of PR_Read. Used by js_FileRead */
785
+ static int32
786
+ js_BufferedRead(JSFile *f, unsigned char *buf, int32 len)
787
+ {
788
+ int32 count = 0;
789
+
790
+ while (f->nbBytesInBuf>0&&len>0) {
791
+ buf[0] = f->byteBuffer[0];
792
+ f->byteBuffer[0] = f->byteBuffer[1];
793
+ f->byteBuffer[1] = f->byteBuffer[2];
794
+ f->nbBytesInBuf--;
795
+ len--;
796
+ buf+=1;
797
+ count++;
798
+ }
799
+
800
+ if (len > 0) {
801
+ count += (!f->isNative)
802
+ ? PR_Read(f->handle, buf, len)
803
+ : fread(buf, 1, len, f->nativehandle);
804
+ }
805
+ return count;
806
+ }
807
+
808
+ static int32
809
+ js_FileRead(JSContext *cx, JSFile *file, jschar *buf, int32 len, int32 mode)
810
+ {
811
+ unsigned char *aux;
812
+ int32 count = 0, i;
813
+ jsint remainder;
814
+ unsigned char utfbuf[3];
815
+
816
+ if (file->charBufferUsed) {
817
+ buf[0] = file->charBuffer;
818
+ buf++;
819
+ len--;
820
+ file->charBufferUsed = JS_FALSE;
821
+ }
822
+
823
+ switch (mode) {
824
+ case ASCII:
825
+ aux = (unsigned char*)JS_malloc(cx, len);
826
+ if (!aux)
827
+ return 0;
828
+
829
+ count = js_BufferedRead(file, aux, len);
830
+ if (count == -1) {
831
+ JS_free(cx, aux);
832
+ return 0;
833
+ }
834
+
835
+ for (i = 0; i < len; i++)
836
+ buf[i] = (jschar)aux[i];
837
+
838
+ JS_free(cx, aux);
839
+ break;
840
+
841
+ case UTF8:
842
+ remainder = 0;
843
+ for (count = 0;count<len;count++) {
844
+ i = js_BufferedRead(file, utfbuf+remainder, 3-remainder);
845
+ if (i<=0) {
846
+ return count;
847
+ }
848
+ i = utf8_to_ucs2_char(utfbuf, (int16)i, &buf[count] );
849
+ if (i<0) {
850
+ return count;
851
+ } else {
852
+ if (i==1) {
853
+ utfbuf[0] = utfbuf[1];
854
+ utfbuf[1] = utfbuf[2];
855
+ remainder = 2;
856
+ } else if (i==2) {
857
+ utfbuf[0] = utfbuf[2];
858
+ remainder = 1;
859
+ } else if (i==3) {
860
+ remainder = 0;
861
+ }
862
+ }
863
+ }
864
+ while (remainder>0) {
865
+ file->byteBuffer[file->nbBytesInBuf] = utfbuf[0];
866
+ file->nbBytesInBuf++;
867
+ utfbuf[0] = utfbuf[1];
868
+ utfbuf[1] = utfbuf[2];
869
+ remainder--;
870
+ }
871
+ break;
872
+
873
+ case UCS2:
874
+ count = js_BufferedRead(file, (unsigned char *)buf, len * 2) >> 1;
875
+ if (count == -1)
876
+ return 0;
877
+
878
+ break;
879
+
880
+ default:
881
+ /* Not reached. */
882
+ JS_ASSERT(0);
883
+ }
884
+
885
+ if(count == -1) {
886
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
887
+ JSFILEMSG_OP_FAILED, "read", file->path);
888
+ }
889
+
890
+ return count;
891
+ }
892
+
893
+ static int32
894
+ js_FileSeek(JSContext *cx, JSFile *file, int32 len, int32 mode)
895
+ {
896
+ int32 count = 0, i;
897
+ jsint remainder;
898
+ unsigned char utfbuf[3];
899
+ jschar tmp;
900
+
901
+ switch (mode) {
902
+ case ASCII:
903
+ count = PR_Seek(file->handle, len, PR_SEEK_CUR);
904
+ break;
905
+
906
+ case UTF8:
907
+ remainder = 0;
908
+ for (count = 0;count<len;count++) {
909
+ i = js_BufferedRead(file, utfbuf+remainder, 3-remainder);
910
+ if (i<=0) {
911
+ return 0;
912
+ }
913
+ i = utf8_to_ucs2_char(utfbuf, (int16)i, &tmp );
914
+ if (i<0) {
915
+ return 0;
916
+ } else {
917
+ if (i==1) {
918
+ utfbuf[0] = utfbuf[1];
919
+ utfbuf[1] = utfbuf[2];
920
+ remainder = 2;
921
+ } else if (i==2) {
922
+ utfbuf[0] = utfbuf[2];
923
+ remainder = 1;
924
+ } else if (i==3) {
925
+ remainder = 0;
926
+ }
927
+ }
928
+ }
929
+ while (remainder>0) {
930
+ file->byteBuffer[file->nbBytesInBuf] = utfbuf[0];
931
+ file->nbBytesInBuf++;
932
+ utfbuf[0] = utfbuf[1];
933
+ utfbuf[1] = utfbuf[2];
934
+ remainder--;
935
+ }
936
+ break;
937
+
938
+ case UCS2:
939
+ count = PR_Seek(file->handle, len*2, PR_SEEK_CUR)/2;
940
+ break;
941
+
942
+ default:
943
+ /* Not reached. */
944
+ JS_ASSERT(0);
945
+ }
946
+
947
+ if(count == -1) {
948
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
949
+ JSFILEMSG_OP_FAILED, "seek", file->path);
950
+ }
951
+
952
+ return count;
953
+ }
954
+
955
+ static int32
956
+ js_FileWrite(JSContext *cx, JSFile *file, jschar *buf, int32 len, int32 mode)
957
+ {
958
+ unsigned char *aux;
959
+ int32 count = 0, i, j;
960
+ unsigned char *utfbuf;
961
+
962
+ switch (mode) {
963
+ case ASCII:
964
+ aux = (unsigned char*)JS_malloc(cx, len);
965
+ if (!aux)
966
+ return 0;
967
+
968
+ for (i = 0; i<len; i++)
969
+ aux[i] = buf[i] % 256;
970
+
971
+ count = (!file->isNative)
972
+ ? PR_Write(file->handle, aux, len)
973
+ : fwrite(aux, 1, len, file->nativehandle);
974
+
975
+ if (count==-1) {
976
+ JS_free(cx, aux);
977
+ return 0;
978
+ }
979
+
980
+ JS_free(cx, aux);
981
+ break;
982
+
983
+ case UTF8:
984
+ utfbuf = (unsigned char*)JS_malloc(cx, len*3);
985
+ if (!utfbuf) return 0;
986
+ i = 0;
987
+ for (count = 0;count<len;count++) {
988
+ j = one_ucs2_to_utf8_char(utfbuf+i, utfbuf+len*3, buf[count]);
989
+ if (j==-1) {
990
+ JS_free(cx, utfbuf);
991
+ return 0;
992
+ }
993
+ i+=j;
994
+ }
995
+ j = (!file->isNative)
996
+ ? PR_Write(file->handle, utfbuf, i)
997
+ : fwrite(utfbuf, 1, i, file->nativehandle);
998
+
999
+ if (j<i) {
1000
+ JS_free(cx, utfbuf);
1001
+ return 0;
1002
+ }
1003
+ JS_free(cx, utfbuf);
1004
+ break;
1005
+
1006
+ case UCS2:
1007
+ count = (!file->isNative)
1008
+ ? PR_Write(file->handle, buf, len*2) >> 1
1009
+ : fwrite(buf, 1, len*2, file->nativehandle) >> 1;
1010
+
1011
+ if (count == -1)
1012
+ return 0;
1013
+ break;
1014
+
1015
+ default:
1016
+ /* Not reached. */
1017
+ JS_ASSERT(0);
1018
+ }
1019
+
1020
+ if(count == -1) {
1021
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1022
+ JSFILEMSG_OP_FAILED, "write", file->path);
1023
+ }
1024
+
1025
+ return count;
1026
+ }
1027
+
1028
+ /* ----------------------------- Property checkers -------------------------- */
1029
+ static JSBool
1030
+ js_exists(JSContext *cx, JSFile *file)
1031
+ {
1032
+ if (file->isNative) {
1033
+ /* It doesn't make sense for a pipe of stdstream. */
1034
+ return JS_FALSE;
1035
+ }
1036
+
1037
+ return PR_Access(file->path, PR_ACCESS_EXISTS) == PR_SUCCESS;
1038
+ }
1039
+
1040
+ static JSBool
1041
+ js_canRead(JSContext *cx, JSFile *file)
1042
+ {
1043
+ if (!file->isNative) {
1044
+ if (file->isOpen && !(file->mode & PR_RDONLY))
1045
+ return JS_FALSE;
1046
+ return PR_Access(file->path, PR_ACCESS_READ_OK) == PR_SUCCESS;
1047
+ }
1048
+
1049
+ if (file->isPipe) {
1050
+ /* Is this pipe open for reading? */
1051
+ return file->path[0] == PIPE_SYMBOL;
1052
+ }
1053
+
1054
+ return !strcmp(file->path, STDINPUT_NAME);
1055
+ }
1056
+
1057
+ static JSBool
1058
+ js_canWrite(JSContext *cx, JSFile *file)
1059
+ {
1060
+ if (!file->isNative) {
1061
+ if (file->isOpen && !(file->mode & PR_WRONLY))
1062
+ return JS_FALSE;
1063
+ return PR_Access(file->path, PR_ACCESS_WRITE_OK) == PR_SUCCESS;
1064
+ }
1065
+
1066
+ if(file->isPipe) {
1067
+ /* Is this pipe open for writing? */
1068
+ return file->path[strlen(file->path)-1] == PIPE_SYMBOL;
1069
+ }
1070
+
1071
+ return !strcmp(file->path, STDOUTPUT_NAME) ||
1072
+ !strcmp(file->path, STDERROR_NAME);
1073
+ }
1074
+
1075
+ static JSBool
1076
+ js_isFile(JSContext *cx, JSFile *file)
1077
+ {
1078
+ if (!file->isNative) {
1079
+ PRFileInfo info;
1080
+
1081
+ if (file->isOpen
1082
+ ? PR_GetOpenFileInfo(file->handle, &info)
1083
+ : PR_GetFileInfo(file->path, &info) != PR_SUCCESS) {
1084
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1085
+ JSFILEMSG_CANNOT_ACCESS_FILE_STATUS, file->path);
1086
+ return JS_FALSE;
1087
+ }
1088
+
1089
+ return info.type == PR_FILE_FILE;
1090
+ }
1091
+
1092
+ /* This doesn't make sense for a pipe of stdstream. */
1093
+ return JS_FALSE;
1094
+ }
1095
+
1096
+ static JSBool
1097
+ js_isDirectory(JSContext *cx, JSFile *file)
1098
+ {
1099
+ if(!file->isNative){
1100
+ PRFileInfo info;
1101
+
1102
+ /* Hack needed to get get_property to work. */
1103
+ if (!js_exists(cx, file))
1104
+ return JS_FALSE;
1105
+
1106
+ if (file->isOpen
1107
+ ? PR_GetOpenFileInfo(file->handle, &info)
1108
+ : PR_GetFileInfo(file->path, &info) != PR_SUCCESS) {
1109
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1110
+ JSFILEMSG_CANNOT_ACCESS_FILE_STATUS, file->path);
1111
+ return JS_FALSE;
1112
+ }
1113
+
1114
+ return info.type == PR_FILE_DIRECTORY;
1115
+ }
1116
+
1117
+ /* This doesn't make sense for a pipe of stdstream. */
1118
+ return JS_FALSE;
1119
+ }
1120
+
1121
+ static jsval
1122
+ js_size(JSContext *cx, JSFile *file)
1123
+ {
1124
+ PRFileInfo info;
1125
+
1126
+ JSFILE_CHECK_NATIVE("size");
1127
+
1128
+ if (file->isOpen
1129
+ ? PR_GetOpenFileInfo(file->handle, &info)
1130
+ : PR_GetFileInfo(file->path, &info) != PR_SUCCESS) {
1131
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1132
+ JSFILEMSG_CANNOT_ACCESS_FILE_STATUS, file->path);
1133
+ return JSVAL_VOID;
1134
+ }
1135
+
1136
+ return INT_TO_JSVAL(info.size);
1137
+
1138
+ out:
1139
+ return JSVAL_VOID;
1140
+ }
1141
+
1142
+ /*
1143
+ * Return the parent object
1144
+ */
1145
+ static JSBool
1146
+ js_parent(JSContext *cx, JSFile *file, jsval *resultp)
1147
+ {
1148
+ char *str;
1149
+
1150
+ /* Since we only care about pipes and native files, return NULL. */
1151
+ if (file->isNative) {
1152
+ *resultp = JSVAL_VOID;
1153
+ return JS_TRUE;
1154
+ }
1155
+
1156
+ str = js_fileDirectoryName(cx, file->path);
1157
+ if (!str)
1158
+ return JS_FALSE;
1159
+
1160
+ /* If the directory is equal to the original path, we're at the root. */
1161
+ if (!strcmp(file->path, str)) {
1162
+ *resultp = JSVAL_NULL;
1163
+ } else {
1164
+ JSObject *obj = js_NewFileObject(cx, str);
1165
+ if (!obj) {
1166
+ JS_free(cx, str);
1167
+ return JS_FALSE;
1168
+ }
1169
+ *resultp = OBJECT_TO_JSVAL(obj);
1170
+ }
1171
+
1172
+ JS_free(cx, str);
1173
+ return JS_TRUE;
1174
+ }
1175
+
1176
+ static JSBool
1177
+ js_name(JSContext *cx, JSFile *file, jsval *vp)
1178
+ {
1179
+ char *name;
1180
+ JSString *str;
1181
+
1182
+ if (file->isPipe) {
1183
+ *vp = JSVAL_VOID;
1184
+ return JS_TRUE;
1185
+ }
1186
+
1187
+ name = js_fileBaseName(cx, file->path);
1188
+ if (!name)
1189
+ return JS_FALSE;
1190
+
1191
+ str = JS_NewString(cx, name, strlen(name));
1192
+ if (!str) {
1193
+ JS_free(cx, name);
1194
+ return JS_FALSE;
1195
+ }
1196
+
1197
+ *vp = STRING_TO_JSVAL(str);
1198
+ return JS_TRUE;
1199
+ }
1200
+
1201
+ /* ------------------------------ File object methods ---------------------------- */
1202
+ static JSBool
1203
+ file_open(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1204
+ {
1205
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1206
+ JSString *strmode, *strtype;
1207
+ char *ctype, *mode;
1208
+ int32 mask, type;
1209
+ int len;
1210
+
1211
+ mode = NULL;
1212
+
1213
+ SECURITY_CHECK(cx, NULL, "open", file);
1214
+
1215
+ /* A native file that is already open */
1216
+ if(file->isOpen && file->isNative) {
1217
+ JS_ReportWarning(cx, "Native file %s is already open, proceeding",
1218
+ file->path);
1219
+ goto good;
1220
+ }
1221
+
1222
+ /* Close before proceeding */
1223
+ if (file->isOpen) {
1224
+ JS_ReportWarning(cx, "File %s is already open, we will close it and "
1225
+ "reopen, proceeding", file->path);
1226
+ if(!file_close(cx, obj, 0, NULL, rval))
1227
+ goto out;
1228
+ }
1229
+
1230
+ if (js_isDirectory(cx, file)) {
1231
+ JS_ReportWarning(cx, "%s seems to be a directory, there is no point in "
1232
+ "trying to open it, proceeding", file->path);
1233
+ goto good;
1234
+ }
1235
+
1236
+ /* Path must be defined at this point */
1237
+ len = strlen(file->path);
1238
+
1239
+ /* Mode */
1240
+ if (argc >= 1) {
1241
+ strmode = JS_ValueToString(cx, argv[0]);
1242
+ if (!strmode) {
1243
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1244
+ JSFILEMSG_FIRST_ARGUMENT_OPEN_NOT_STRING_ERROR,
1245
+ argv[0]);
1246
+ goto out;
1247
+ }
1248
+ mode = JS_strdup(cx, JS_GetStringBytes(strmode));
1249
+ } else {
1250
+ if(file->path[0]==PIPE_SYMBOL) {
1251
+ /* pipe default mode */
1252
+ mode = JS_strdup(cx, "read");
1253
+ } else if(file->path[len-1]==PIPE_SYMBOL) {
1254
+ /* pipe default mode */
1255
+ mode = JS_strdup(cx, "write");
1256
+ } else {
1257
+ /* non-destructive, permissive defaults. */
1258
+ mode = JS_strdup(cx, "readWrite,append,create");
1259
+ }
1260
+ }
1261
+
1262
+ /* Process the mode */
1263
+ mask = 0;
1264
+ /* TODO: this is pretty ugly, we walk thru the string too many times */
1265
+ mask |= js_FileHasOption(cx, mode, "read") ? PR_RDONLY : 0;
1266
+ mask |= js_FileHasOption(cx, mode, "write") ? PR_WRONLY : 0;
1267
+ mask |= js_FileHasOption(cx, mode, "readWrite")? PR_RDWR : 0;
1268
+ mask |= js_FileHasOption(cx, mode, "append") ? PR_APPEND : 0;
1269
+ mask |= js_FileHasOption(cx, mode, "create") ? PR_CREATE_FILE : 0;
1270
+ mask |= js_FileHasOption(cx, mode, "replace") ? PR_TRUNCATE : 0;
1271
+
1272
+ if (mask & PR_RDWR)
1273
+ mask |= (PR_RDONLY | PR_WRONLY);
1274
+ if ((mask & PR_RDONLY) && (mask & PR_WRONLY))
1275
+ mask |= PR_RDWR;
1276
+
1277
+ file->hasAutoflush |= js_FileHasOption(cx, mode, "autoflush");
1278
+
1279
+ /* Type */
1280
+ if (argc > 1) {
1281
+ strtype = JS_ValueToString(cx, argv[1]);
1282
+ if (!strtype) {
1283
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1284
+ JSFILEMSG_SECOND_ARGUMENT_OPEN_NOT_STRING_ERROR,
1285
+ argv[1]);
1286
+ goto out;
1287
+ }
1288
+ ctype = JS_GetStringBytes(strtype);
1289
+
1290
+ if(!strcmp(ctype, utfstring)) {
1291
+ type = UTF8;
1292
+ } else if (!strcmp(ctype, unicodestring)) {
1293
+ type = UCS2;
1294
+ } else {
1295
+ if (strcmp(ctype, asciistring)) {
1296
+ JS_ReportWarning(cx, "File type %s is not supported, using "
1297
+ "'text' instead, proceeding", ctype);
1298
+ }
1299
+ type = ASCII;
1300
+ }
1301
+ } else {
1302
+ type = ASCII;
1303
+ }
1304
+
1305
+ /* Save the relevant fields */
1306
+ file->type = type;
1307
+ file->mode = mask;
1308
+ file->nativehandle = NULL;
1309
+ file->hasRandomAccess = (type != UTF8);
1310
+
1311
+ /*
1312
+ * Deal with pipes here. We can't use NSPR for pipes, so we have to use
1313
+ * POPEN.
1314
+ */
1315
+ if (file->path[0]==PIPE_SYMBOL || file->path[len-1]==PIPE_SYMBOL) {
1316
+ if (file->path[0] == PIPE_SYMBOL && file->path[len-1] == PIPE_SYMBOL) {
1317
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1318
+ JSFILEMSG_BIDIRECTIONAL_PIPE_NOT_SUPPORTED);
1319
+ goto out;
1320
+ } else {
1321
+ int i = 0;
1322
+ char pipemode[3];
1323
+ SECURITY_CHECK(cx, NULL, "pipe_open", file);
1324
+
1325
+ if(file->path[0] == PIPE_SYMBOL){
1326
+ if(mask & (PR_WRONLY | PR_APPEND | PR_CREATE_FILE | PR_TRUNCATE)){
1327
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1328
+ JSFILEMSG_OPEN_MODE_NOT_SUPPORTED_WITH_PIPES,
1329
+ mode, file->path);
1330
+ goto out;
1331
+ }
1332
+ /* open(SPOOLER, "| cat -v | lpr -h 2>/dev/null") -- pipe for writing */
1333
+ pipemode[i++] = 'r';
1334
+ #ifndef XP_UNIX
1335
+ pipemode[i++] = file->type==UTF8 ? 'b' : 't';
1336
+ #endif
1337
+ pipemode[i++] = '\0';
1338
+ file->nativehandle = POPEN(&file->path[1], pipemode);
1339
+ } else if(file->path[len-1] == PIPE_SYMBOL) {
1340
+ char *command = JS_malloc(cx, len);
1341
+
1342
+ strncpy(command, file->path, len-1);
1343
+ command[len-1] = '\0';
1344
+ /* open(STATUS, "netstat -an 2>&1 |") */
1345
+ pipemode[i++] = 'w';
1346
+ #ifndef XP_UNIX
1347
+ pipemode[i++] = file->type==UTF8 ? 'b' : 't';
1348
+ #endif
1349
+ pipemode[i++] = '\0';
1350
+ file->nativehandle = POPEN(command, pipemode);
1351
+ JS_free(cx, command);
1352
+ }
1353
+ /* set the flags */
1354
+ file->isNative = JS_TRUE;
1355
+ file->isPipe = JS_TRUE;
1356
+ file->hasRandomAccess = JS_FALSE;
1357
+ }
1358
+ } else {
1359
+ /* TODO: what about the permissions?? Java ignores the problem... */
1360
+ file->handle = PR_Open(file->path, mask, 0644);
1361
+ }
1362
+
1363
+ js_ResetBuffers(file);
1364
+ JS_free(cx, mode);
1365
+ mode = NULL;
1366
+
1367
+ /* Set the open flag and return result */
1368
+ if (file->handle == NULL && file->nativehandle == NULL) {
1369
+ file->isOpen = JS_FALSE;
1370
+
1371
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1372
+ JSFILEMSG_OP_FAILED, "open", file->path);
1373
+ goto out;
1374
+ }
1375
+
1376
+ good:
1377
+ file->isOpen = JS_TRUE;
1378
+ *rval = JSVAL_TRUE;
1379
+ return JS_TRUE;
1380
+
1381
+ out:
1382
+ if(mode)
1383
+ JS_free(cx, mode);
1384
+ return JS_FALSE;
1385
+ }
1386
+
1387
+ static JSBool
1388
+ file_close(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1389
+ {
1390
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1391
+
1392
+ SECURITY_CHECK(cx, NULL, "close", file);
1393
+
1394
+ if(!file->isOpen){
1395
+ JS_ReportWarning(cx, "File %s is not open, can't close it, proceeding",
1396
+ file->path);
1397
+ goto out;
1398
+ }
1399
+
1400
+ if(!file->isPipe){
1401
+ if(file->isNative){
1402
+ JS_ReportWarning(cx, "Unable to close a native file, proceeding", file->path);
1403
+ goto out;
1404
+ }else{
1405
+ if(file->handle && PR_Close(file->handle)){
1406
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1407
+ JSFILEMSG_OP_FAILED, "close", file->path);
1408
+
1409
+ goto out;
1410
+ }
1411
+ }
1412
+ }else{
1413
+ if(PCLOSE(file->nativehandle)==-1){
1414
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1415
+ JSFILEMSG_OP_FAILED, "pclose", file->path);
1416
+ goto out;
1417
+ }
1418
+ }
1419
+
1420
+ js_ResetAttributes(file);
1421
+ *rval = JSVAL_TRUE;
1422
+ return JS_TRUE;
1423
+
1424
+ out:
1425
+ return JS_FALSE;
1426
+ }
1427
+
1428
+
1429
+ static JSBool
1430
+ file_remove(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1431
+ {
1432
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1433
+
1434
+ SECURITY_CHECK(cx, NULL, "remove", file);
1435
+ JSFILE_CHECK_NATIVE("remove");
1436
+ JSFILE_CHECK_CLOSED("remove");
1437
+
1438
+ if ((js_isDirectory(cx, file) ?
1439
+ PR_RmDir(file->path) : PR_Delete(file->path))==PR_SUCCESS) {
1440
+ js_ResetAttributes(file);
1441
+ *rval = JSVAL_TRUE;
1442
+ return JS_TRUE;
1443
+ } else {
1444
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1445
+ JSFILEMSG_OP_FAILED, "remove", file->path);
1446
+ goto out;
1447
+ }
1448
+ out:
1449
+ *rval = JSVAL_FALSE;
1450
+ return JS_FALSE;
1451
+ }
1452
+
1453
+ /* Raw PR-based function. No text processing. Just raw data copying. */
1454
+ static JSBool
1455
+ file_copyTo(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1456
+ {
1457
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1458
+ char *dest = NULL;
1459
+ PRFileDesc *handle = NULL;
1460
+ char *buffer;
1461
+ jsval count, size;
1462
+ JSBool fileInitiallyOpen=JS_FALSE;
1463
+
1464
+ SECURITY_CHECK(cx, NULL, "copyTo", file); /* may need a second argument!*/
1465
+ JSFILE_CHECK_ONE_ARG("copyTo");
1466
+ JSFILE_CHECK_NATIVE("copyTo");
1467
+ /* remeber the state */
1468
+ fileInitiallyOpen = file->isOpen;
1469
+ JSFILE_CHECK_READ;
1470
+
1471
+ dest = JS_GetStringBytes(JS_ValueToString(cx, argv[0]));
1472
+
1473
+ /* make sure we are not reading a file open for writing */
1474
+ if (file->isOpen && !js_canRead(cx, file)) {
1475
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1476
+ JSFILEMSG_CANNOT_COPY_FILE_OPEN_FOR_WRITING_ERROR, file->path);
1477
+ goto out;
1478
+ }
1479
+
1480
+ if (file->handle==NULL){
1481
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1482
+ JSFILEMSG_OP_FAILED, "open", file->path);
1483
+ goto out;
1484
+ }
1485
+
1486
+ handle = PR_Open(dest, PR_WRONLY|PR_CREATE_FILE|PR_TRUNCATE, 0644);
1487
+
1488
+ if(!handle){
1489
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1490
+ JSFILEMSG_OP_FAILED, "open", dest);
1491
+ goto out;
1492
+ }
1493
+
1494
+ if ((size=js_size(cx, file))==JSVAL_VOID) {
1495
+ goto out;
1496
+ }
1497
+
1498
+ buffer = JS_malloc(cx, size);
1499
+
1500
+ count = INT_TO_JSVAL(PR_Read(file->handle, buffer, size));
1501
+
1502
+ /* reading panic */
1503
+ if (count!=size) {
1504
+ JS_free(cx, buffer);
1505
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1506
+ JSFILEMSG_COPY_READ_ERROR, file->path);
1507
+ goto out;
1508
+ }
1509
+
1510
+ count = INT_TO_JSVAL(PR_Write(handle, buffer, JSVAL_TO_INT(size)));
1511
+
1512
+ /* writing panic */
1513
+ if (count!=size) {
1514
+ JS_free(cx, buffer);
1515
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1516
+ JSFILEMSG_COPY_WRITE_ERROR, file->path);
1517
+ goto out;
1518
+ }
1519
+
1520
+ JS_free(cx, buffer);
1521
+
1522
+ if(!fileInitiallyOpen){
1523
+ if(!file_close(cx, obj, 0, NULL, rval)) goto out;
1524
+ }
1525
+
1526
+ if(PR_Close(handle)!=PR_SUCCESS){
1527
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1528
+ JSFILEMSG_OP_FAILED, "close", dest);
1529
+ goto out;
1530
+ }
1531
+
1532
+ *rval = JSVAL_TRUE;
1533
+ return JS_TRUE;
1534
+ out:
1535
+ if(file->isOpen && !fileInitiallyOpen){
1536
+ if(PR_Close(file->handle)!=PR_SUCCESS){
1537
+ JS_ReportWarning(cx, "Can't close %s, proceeding", file->path);
1538
+ }
1539
+ }
1540
+
1541
+ if(handle && PR_Close(handle)!=PR_SUCCESS){
1542
+ JS_ReportWarning(cx, "Can't close %s, proceeding", dest);
1543
+ }
1544
+
1545
+ *rval = JSVAL_FALSE;
1546
+ return JS_FALSE;
1547
+ }
1548
+
1549
+ static JSBool
1550
+ file_renameTo(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1551
+ {
1552
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1553
+ char *dest;
1554
+
1555
+ SECURITY_CHECK(cx, NULL, "renameTo", file); /* may need a second argument!*/
1556
+ JSFILE_CHECK_ONE_ARG("renameTo");
1557
+ JSFILE_CHECK_NATIVE("renameTo");
1558
+ JSFILE_CHECK_CLOSED("renameTo");
1559
+
1560
+ dest = RESOLVE_PATH(cx, JS_GetStringBytes(JS_ValueToString(cx, argv[0])));
1561
+
1562
+ if (PR_Rename(file->path, dest)==PR_SUCCESS){
1563
+ /* copy the new filename */
1564
+ JS_free(cx, file->path);
1565
+ file->path = dest;
1566
+ *rval = JSVAL_TRUE;
1567
+ return JS_TRUE;
1568
+ }else{
1569
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1570
+ JSFILEMSG_RENAME_FAILED, file->path, dest);
1571
+ goto out;
1572
+ }
1573
+ out:
1574
+ *rval = JSVAL_FALSE;
1575
+ return JS_FALSE;
1576
+ }
1577
+
1578
+ static JSBool
1579
+ file_flush(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1580
+ {
1581
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1582
+
1583
+ SECURITY_CHECK(cx, NULL, "flush", file);
1584
+ JSFILE_CHECK_NATIVE("flush");
1585
+ JSFILE_CHECK_OPEN("flush");
1586
+
1587
+ if (PR_Sync(file->handle)==PR_SUCCESS){
1588
+ *rval = JSVAL_TRUE;
1589
+ return JS_TRUE;
1590
+ }else{
1591
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1592
+ JSFILEMSG_OP_FAILED, "flush", file->path);
1593
+ goto out;
1594
+ }
1595
+ out:
1596
+ *rval = JSVAL_FALSE;
1597
+ return JS_FALSE;
1598
+ }
1599
+
1600
+ static JSBool
1601
+ file_write(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1602
+ {
1603
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1604
+ JSString *str;
1605
+ int32 count;
1606
+ uintN i;
1607
+
1608
+ SECURITY_CHECK(cx, NULL, "write", file);
1609
+ JSFILE_CHECK_WRITE;
1610
+
1611
+ for (i = 0; i<argc; i++) {
1612
+ str = JS_ValueToString(cx, argv[i]);
1613
+ count = js_FileWrite(cx, file, JS_GetStringChars(str),
1614
+ JS_GetStringLength(str), file->type);
1615
+ if (count==-1){
1616
+ *rval = JSVAL_FALSE;
1617
+ return JS_FALSE;
1618
+ }
1619
+ }
1620
+
1621
+ *rval = JSVAL_TRUE;
1622
+ return JS_TRUE;
1623
+ out:
1624
+ *rval = JSVAL_FALSE;
1625
+ return JS_FALSE;
1626
+ }
1627
+
1628
+ static JSBool
1629
+ file_writeln(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1630
+ {
1631
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1632
+ JSString *str;
1633
+
1634
+ SECURITY_CHECK(cx, NULL, "writeln", file);
1635
+ JSFILE_CHECK_WRITE;
1636
+
1637
+ /* don't report an error here */
1638
+ if(!file_write(cx, obj, argc, argv, rval)) return JS_FALSE;
1639
+ /* don't do security here -- we passed the check in file_write */
1640
+ str = JS_NewStringCopyZ(cx, "\n");
1641
+
1642
+ if (js_FileWrite(cx, file, JS_GetStringChars(str), JS_GetStringLength(str),
1643
+ file->type)==-1){
1644
+ *rval = JSVAL_FALSE;
1645
+ return JS_FALSE;
1646
+ }
1647
+
1648
+ /* eol causes flush if hasAutoflush is turned on */
1649
+ if (file->hasAutoflush)
1650
+ file_flush(cx, obj, 0, NULL, rval);
1651
+
1652
+ *rval = JSVAL_TRUE;
1653
+ return JS_TRUE;
1654
+ out:
1655
+ *rval = JSVAL_FALSE;
1656
+ return JS_FALSE;
1657
+ }
1658
+
1659
+ static JSBool
1660
+ file_writeAll(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1661
+ {
1662
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1663
+ jsuint i;
1664
+ jsuint limit;
1665
+ JSObject *array;
1666
+ JSObject *elem;
1667
+ jsval elemval;
1668
+
1669
+ SECURITY_CHECK(cx, NULL, "writeAll", file);
1670
+ JSFILE_CHECK_ONE_ARG("writeAll");
1671
+ JSFILE_CHECK_WRITE;
1672
+
1673
+ if (!JS_IsArrayObject(cx, JSVAL_TO_OBJECT(argv[0]))) {
1674
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1675
+ JSFILEMSG_FIRST_ARGUMENT_WRITEALL_NOT_ARRAY_ERROR);
1676
+ goto out;
1677
+ }
1678
+
1679
+ array = JSVAL_TO_OBJECT(argv[0]);
1680
+
1681
+ JS_GetArrayLength(cx, array, &limit);
1682
+
1683
+ for (i = 0; i<limit; i++) {
1684
+ if (!JS_GetElement(cx, array, i, &elemval)) return JS_FALSE;
1685
+ elem = JSVAL_TO_OBJECT(elemval);
1686
+ file_writeln(cx, obj, 1, &elemval, rval);
1687
+ }
1688
+
1689
+ *rval = JSVAL_TRUE;
1690
+ return JS_TRUE;
1691
+ out:
1692
+ *rval = JSVAL_FALSE;
1693
+ return JS_FALSE;
1694
+ }
1695
+
1696
+ static JSBool
1697
+ file_read(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1698
+ {
1699
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1700
+ JSString *str;
1701
+ int32 want, count;
1702
+ jschar *buf;
1703
+
1704
+ SECURITY_CHECK(cx, NULL, "read", file);
1705
+ JSFILE_CHECK_ONE_ARG("read");
1706
+ JSFILE_CHECK_READ;
1707
+
1708
+ if (!JS_ValueToInt32(cx, argv[0], &want)){
1709
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1710
+ JSFILEMSG_FIRST_ARGUMENT_MUST_BE_A_NUMBER, "read", argv[0]);
1711
+ goto out;
1712
+ }
1713
+
1714
+ /* want = (want>262144)?262144:want; * arbitrary size limitation */
1715
+
1716
+ buf = JS_malloc(cx, want*sizeof buf[0]);
1717
+ if (!buf) goto out;
1718
+
1719
+ count = js_FileRead(cx, file, buf, want, file->type);
1720
+ if (count>0) {
1721
+ str = JS_NewUCStringCopyN(cx, buf, count);
1722
+ *rval = STRING_TO_JSVAL(str);
1723
+ JS_free(cx, buf);
1724
+ return JS_TRUE;
1725
+ } else {
1726
+ JS_free(cx, buf);
1727
+ goto out;
1728
+ }
1729
+ out:
1730
+ *rval = JSVAL_FALSE;
1731
+ return JS_FALSE;
1732
+ }
1733
+
1734
+ static JSBool
1735
+ file_readln(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1736
+ {
1737
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1738
+ JSString *str;
1739
+ jschar *buf = NULL, *tmp;
1740
+ int32 offset, read;
1741
+ intN room;
1742
+ jschar data, data2;
1743
+
1744
+ SECURITY_CHECK(cx, NULL, "readln", file);
1745
+ JSFILE_CHECK_READ;
1746
+
1747
+ buf = JS_malloc(cx, MAX_LINE_LENGTH * sizeof data);
1748
+ if (!buf)
1749
+ return JS_FALSE;
1750
+
1751
+ room = MAX_LINE_LENGTH - 1;
1752
+ offset = 0;
1753
+
1754
+ for (;;) {
1755
+ read = js_FileRead(cx, file, &data, 1, file->type);
1756
+ if (read < 0)
1757
+ goto out;
1758
+ if (read == 0)
1759
+ goto eof;
1760
+
1761
+ switch (data) {
1762
+ case '\r':
1763
+ read = js_FileRead(cx, file, &data2, 1, file->type);
1764
+ if (read < 0)
1765
+ goto out;
1766
+
1767
+ if (read == 1 && data2 != '\n') {
1768
+ /* We read one char too far. Buffer it. */
1769
+ file->charBuffer = data2;
1770
+ file->charBufferUsed = JS_TRUE;
1771
+ }
1772
+
1773
+ /* Fall through. */
1774
+ case '\n':
1775
+ goto done;
1776
+
1777
+ default:
1778
+ if (--room < 0) {
1779
+ tmp = JS_realloc(cx, buf,
1780
+ (offset + MAX_LINE_LENGTH) * sizeof data);
1781
+ if (!tmp)
1782
+ goto out;
1783
+
1784
+ room = MAX_LINE_LENGTH - 1;
1785
+ buf = tmp;
1786
+ }
1787
+
1788
+ buf[offset++] = data;
1789
+ break;
1790
+ }
1791
+ }
1792
+
1793
+ eof:
1794
+ if (offset == 0) {
1795
+ *rval = JSVAL_NULL;
1796
+ return JS_TRUE;
1797
+ }
1798
+
1799
+ done:
1800
+ buf[offset] = 0;
1801
+ tmp = JS_realloc(cx, buf, (offset + 1) * sizeof data);
1802
+ if (!tmp)
1803
+ goto out;
1804
+
1805
+ str = JS_NewUCString(cx, tmp, offset);
1806
+ if (!str)
1807
+ goto out;
1808
+
1809
+ *rval = STRING_TO_JSVAL(str);
1810
+ return JS_TRUE;
1811
+
1812
+ out:
1813
+ if (buf)
1814
+ JS_free(cx, buf);
1815
+
1816
+ return JS_FALSE;
1817
+ }
1818
+
1819
+ static JSBool
1820
+ file_readAll(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1821
+ {
1822
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1823
+ JSObject *array;
1824
+ jsint len;
1825
+ jsval line;
1826
+ JSBool lineok = JS_FALSE;
1827
+
1828
+ SECURITY_CHECK(cx, NULL, "readAll", file);
1829
+ JSFILE_CHECK_READ;
1830
+
1831
+ array = JS_NewArrayObject(cx, 0, NULL);
1832
+ if (!array)
1833
+ return JS_FALSE;
1834
+ *rval = OBJECT_TO_JSVAL(array);
1835
+
1836
+ len = 0;
1837
+
1838
+ lineok = file_readln(cx, obj, 0, NULL, &line);
1839
+ while (lineok && !JSVAL_IS_NULL(line)) {
1840
+ JS_SetElement(cx, array, len++, &line);
1841
+ lineok = file_readln(cx, obj, 0, NULL, &line);
1842
+ }
1843
+
1844
+ out:
1845
+ return lineok;
1846
+ }
1847
+
1848
+ static JSBool
1849
+ file_seek(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1850
+ {
1851
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1852
+ int32 toskip;
1853
+ int32 pos;
1854
+
1855
+ SECURITY_CHECK(cx, NULL, "seek", file);
1856
+ JSFILE_CHECK_ONE_ARG("seek");
1857
+ JSFILE_CHECK_NATIVE("seek");
1858
+ JSFILE_CHECK_READ;
1859
+
1860
+ if (!JS_ValueToInt32(cx, argv[0], &toskip)){
1861
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1862
+ JSFILEMSG_FIRST_ARGUMENT_MUST_BE_A_NUMBER, "seek", argv[0]);
1863
+ goto out;
1864
+ }
1865
+
1866
+ if(!file->hasRandomAccess){
1867
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1868
+ JSFILEMSG_NO_RANDOM_ACCESS, file->path);
1869
+ goto out;
1870
+ }
1871
+
1872
+ if(js_isDirectory(cx, file)){
1873
+ JS_ReportWarning(cx,"Seek on directories is not supported, proceeding");
1874
+ goto out;
1875
+ }
1876
+
1877
+ pos = js_FileSeek(cx, file, toskip, file->type);
1878
+
1879
+ if (pos!=-1) {
1880
+ *rval = INT_TO_JSVAL(pos);
1881
+ return JS_TRUE;
1882
+ }
1883
+ out:
1884
+ *rval = JSVAL_VOID;
1885
+ return JS_FALSE;
1886
+ }
1887
+
1888
+ static JSBool
1889
+ file_list(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1890
+ {
1891
+ PRDir *dir;
1892
+ PRDirEntry *entry;
1893
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1894
+ JSObject *array;
1895
+ JSObject *eachFile;
1896
+ jsint len;
1897
+ jsval v;
1898
+ JSRegExp *re = NULL;
1899
+ JSFunction *func = NULL;
1900
+ JSString *str;
1901
+ jsval args[1];
1902
+ char *filePath;
1903
+
1904
+ SECURITY_CHECK(cx, NULL, "list", file);
1905
+ JSFILE_CHECK_NATIVE("list");
1906
+
1907
+ if (argc==1) {
1908
+ if (JSVAL_IS_REGEXP(cx, argv[0])) {
1909
+ re = JS_GetPrivate(cx, JSVAL_TO_OBJECT(argv[0]));
1910
+ }else
1911
+ if (VALUE_IS_FUNCTION(cx, argv[0])) {
1912
+ func = JS_GetPrivate(cx, JSVAL_TO_OBJECT(argv[0]));
1913
+ }else{
1914
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1915
+ JSFILEMSG_FIRST_ARGUMENT_MUST_BE_A_FUNCTION_OR_REGEX, argv[0]);
1916
+ goto out;
1917
+ }
1918
+ }
1919
+
1920
+ if (!js_isDirectory(cx, file)) {
1921
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1922
+ JSFILEMSG_CANNOT_DO_LIST_ON_A_FILE, file->path);
1923
+ goto out;
1924
+ }
1925
+
1926
+ dir = PR_OpenDir(file->path);
1927
+ if(!dir){
1928
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1929
+ JSFILEMSG_OP_FAILED, "open", file->path);
1930
+ goto out;
1931
+ }
1932
+
1933
+ /* create JSArray here... */
1934
+ array = JS_NewArrayObject(cx, 0, NULL);
1935
+ len = 0;
1936
+
1937
+ while ((entry = PR_ReadDir(dir, PR_SKIP_BOTH))!=NULL) {
1938
+ /* first, check if we have a regexp */
1939
+ if (re!=NULL) {
1940
+ size_t index = 0;
1941
+
1942
+ str = JS_NewStringCopyZ(cx, entry->name);
1943
+ if(!js_ExecuteRegExp(cx, re, str, &index, JS_TRUE, &v)){
1944
+ /* don't report anything here */
1945
+ goto out;
1946
+ }
1947
+ /* not matched! */
1948
+ if (JSVAL_IS_NULL(v)) {
1949
+ continue;
1950
+ }
1951
+ }else
1952
+ if (func!=NULL) {
1953
+ str = JS_NewStringCopyZ(cx, entry->name);
1954
+ args[0] = STRING_TO_JSVAL(str);
1955
+ if(!JS_CallFunction(cx, obj, func, 1, args, &v)){
1956
+ goto out;
1957
+ }
1958
+
1959
+ if (v==JSVAL_FALSE) {
1960
+ continue;
1961
+ }
1962
+ }
1963
+
1964
+ filePath = js_combinePath(cx, file->path, (char*)entry->name);
1965
+
1966
+ eachFile = js_NewFileObject(cx, filePath);
1967
+ JS_free(cx, filePath);
1968
+ if (!eachFile){
1969
+ JS_ReportWarning(cx, "File %s cannot be retrieved", filePath);
1970
+ continue;
1971
+ }
1972
+ v = OBJECT_TO_JSVAL(eachFile);
1973
+ JS_SetElement(cx, array, len, &v);
1974
+ JS_SetProperty(cx, array, entry->name, &v);
1975
+ len++;
1976
+ }
1977
+
1978
+ if(PR_CloseDir(dir)!=PR_SUCCESS){
1979
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
1980
+ JSFILEMSG_OP_FAILED, "close", file->path);
1981
+ goto out;
1982
+ }
1983
+ *rval = OBJECT_TO_JSVAL(array);
1984
+ return JS_TRUE;
1985
+ out:
1986
+ *rval = JSVAL_NULL;
1987
+ return JS_FALSE;
1988
+ }
1989
+
1990
+ static JSBool
1991
+ file_mkdir(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
1992
+ {
1993
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
1994
+
1995
+ SECURITY_CHECK(cx, NULL, "mkdir", file);
1996
+ JSFILE_CHECK_ONE_ARG("mkdir");
1997
+ JSFILE_CHECK_NATIVE("mkdir");
1998
+
1999
+ /* if the current file is not a directory, find out the directory name */
2000
+ if (!js_isDirectory(cx, file)) {
2001
+ char *dir = js_fileDirectoryName(cx, file->path);
2002
+ JSObject *dirObj = js_NewFileObject(cx, dir);
2003
+
2004
+ JS_free(cx, dir);
2005
+
2006
+ /* call file_mkdir with the right set of parameters if needed */
2007
+ if (file_mkdir(cx, dirObj, argc, argv, rval))
2008
+ return JS_TRUE;
2009
+ else
2010
+ goto out;
2011
+ }else{
2012
+ char *dirName = JS_GetStringBytes(JS_ValueToString(cx, argv[0]));
2013
+ char *fullName;
2014
+
2015
+ fullName = js_combinePath(cx, file->path, dirName);
2016
+ if (PR_MkDir(fullName, 0755)==PR_SUCCESS){
2017
+ *rval = JSVAL_TRUE;
2018
+ JS_free(cx, fullName);
2019
+ return JS_TRUE;
2020
+ }else{
2021
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2022
+ JSFILEMSG_OP_FAILED, "mkdir", fullName);
2023
+ JS_free(cx, fullName);
2024
+ goto out;
2025
+ }
2026
+ }
2027
+ out:
2028
+ *rval = JSVAL_FALSE;
2029
+ return JS_FALSE;
2030
+ }
2031
+
2032
+ static JSBool
2033
+ file_toString(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval*rval)
2034
+ {
2035
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
2036
+ JSString *str;
2037
+
2038
+ str = JS_NewStringCopyZ(cx, file->path);
2039
+ if (!str)
2040
+ return JS_FALSE;
2041
+ *rval = STRING_TO_JSVAL(str);
2042
+ return JS_TRUE;
2043
+ }
2044
+
2045
+ static JSBool
2046
+ file_toURL(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval)
2047
+ {
2048
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
2049
+ char url[MAX_PATH_LENGTH];
2050
+ jschar *urlChars;
2051
+ size_t len;
2052
+ JSString *str;
2053
+
2054
+ JSFILE_CHECK_NATIVE("toURL");
2055
+
2056
+ sprintf(url, "file://%s", file->path);
2057
+
2058
+ len = strlen(url);
2059
+ urlChars = js_InflateString(cx, url, &len);
2060
+ if (!urlChars)
2061
+ return JS_FALSE;
2062
+ str = js_NewString(cx, urlChars, len);
2063
+ if (!str) {
2064
+ JS_free(cx, urlChars);
2065
+ return JS_FALSE;
2066
+ }
2067
+ *rval = STRING_TO_JSVAL(str);
2068
+
2069
+ /* TODO: js_escape in jsstr.h may go away at some point */
2070
+ return js_str_escape(cx, obj, 0, rval, rval);
2071
+
2072
+ out:
2073
+ *rval = JSVAL_VOID;
2074
+ return JS_FALSE;
2075
+ }
2076
+
2077
+
2078
+ static void
2079
+ file_finalize(JSContext *cx, JSObject *obj)
2080
+ {
2081
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
2082
+
2083
+ if(file) {
2084
+ /* Close the file before exiting. */
2085
+ if(file->isOpen && !file->isNative) {
2086
+ jsval vp;
2087
+ file_close(cx, obj, 0, NULL, &vp);
2088
+ }
2089
+
2090
+ if (file->path)
2091
+ JS_free(cx, file->path);
2092
+
2093
+ JS_free(cx, file);
2094
+ }
2095
+ }
2096
+
2097
+ /*
2098
+ Allocates memory for the file object, sets fields to defaults.
2099
+ */
2100
+ static JSFile*
2101
+ file_init(JSContext *cx, JSObject *obj, char *bytes)
2102
+ {
2103
+ JSFile *file;
2104
+
2105
+ file = JS_malloc(cx, sizeof *file);
2106
+ if (!file)
2107
+ return NULL;
2108
+ memset(file, 0 , sizeof *file);
2109
+
2110
+ js_ResetAttributes(file);
2111
+
2112
+ file->path = RESOLVE_PATH(cx, bytes);
2113
+
2114
+ if (!JS_SetPrivate(cx, obj, file)) {
2115
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2116
+ JSFILEMSG_CANNOT_SET_PRIVATE_FILE, file->path);
2117
+ JS_free(cx, file);
2118
+ return NULL;
2119
+ }
2120
+
2121
+ return file;
2122
+ }
2123
+
2124
+ /* Returns a JSObject. This function is globally visible */
2125
+ JS_PUBLIC_API(JSObject*)
2126
+ js_NewFileObject(JSContext *cx, char *filename)
2127
+ {
2128
+ JSObject *obj;
2129
+ JSFile *file;
2130
+
2131
+ obj = JS_NewObject(cx, &js_FileClass, NULL, NULL);
2132
+ if (!obj){
2133
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2134
+ JSFILEMSG_OBJECT_CREATION_FAILED, "js_NewFileObject");
2135
+ return NULL;
2136
+ }
2137
+ file = file_init(cx, obj, filename);
2138
+ if(!file) return NULL;
2139
+ return obj;
2140
+ }
2141
+
2142
+ /* Internal function, used for cases which NSPR file support doesn't cover */
2143
+ JSObject*
2144
+ js_NewFileObjectFromFILE(JSContext *cx, FILE *nativehandle, char *filename,
2145
+ int32 mode, JSBool open, JSBool randomAccess)
2146
+ {
2147
+ JSObject *obj;
2148
+ JSFile *file;
2149
+
2150
+ obj = JS_NewObject(cx, &js_FileClass, NULL, NULL);
2151
+ if (!obj){
2152
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2153
+ JSFILEMSG_OBJECT_CREATION_FAILED, "js_NewFileObjectFromFILE");
2154
+ return NULL;
2155
+ }
2156
+ file = file_init(cx, obj, filename);
2157
+ if(!file) return NULL;
2158
+
2159
+ file->nativehandle = nativehandle;
2160
+
2161
+ /* free result of RESOLVE_PATH from file_init. */
2162
+ JS_ASSERT(file->path != NULL);
2163
+ JS_free(cx, file->path);
2164
+
2165
+ file->path = strdup(filename);
2166
+ file->isOpen = open;
2167
+ file->mode = mode;
2168
+ file->hasRandomAccess = randomAccess;
2169
+ file->isNative = JS_TRUE;
2170
+ return obj;
2171
+ }
2172
+
2173
+ /*
2174
+ Real file constructor that is called from JavaScript.
2175
+ Basically, does error processing and calls file_init.
2176
+ */
2177
+ static JSBool
2178
+ file_constructor(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
2179
+ jsval *rval)
2180
+ {
2181
+ JSString *str;
2182
+ JSFile *file;
2183
+
2184
+ if (!(cx->fp->flags & JSFRAME_CONSTRUCTING)) {
2185
+ /* Replace obj with a new File object. */
2186
+ obj = JS_NewObject(cx, &js_FileClass, NULL, NULL);
2187
+ if (!obj)
2188
+ return JS_FALSE;
2189
+ *rval = OBJECT_TO_JSVAL(obj);
2190
+ }
2191
+
2192
+ str = (argc == 0)
2193
+ ? JS_InternString(cx, "")
2194
+ : JS_ValueToString(cx, argv[0]);
2195
+
2196
+ if (!str) {
2197
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2198
+ JSFILEMSG_FIRST_ARGUMENT_CONSTRUCTOR_NOT_STRING_ERROR,
2199
+ argv[0]);
2200
+ return JS_FALSE;
2201
+ }
2202
+
2203
+ file = file_init(cx, obj, JS_GetStringBytes(str));
2204
+ if (!file)
2205
+ return JS_FALSE;
2206
+
2207
+ SECURITY_CHECK(cx, NULL, "constructor", file);
2208
+
2209
+ return JS_TRUE;
2210
+ }
2211
+
2212
+ /* -------------------- File methods and properties ------------------------- */
2213
+ static JSFunctionSpec file_functions[] = {
2214
+ { "open", file_open, 0},
2215
+ { "close", file_close, 0},
2216
+ { "remove", file_remove, 0},
2217
+ { "copyTo", file_copyTo, 0},
2218
+ { "renameTo", file_renameTo, 0},
2219
+ { "flush", file_flush, 0},
2220
+ { "seek", file_seek, 0},
2221
+ { "read", file_read, 0},
2222
+ { "readln", file_readln, 0},
2223
+ { "readAll", file_readAll, 0},
2224
+ { "write", file_write, 0},
2225
+ { "writeln", file_writeln, 0},
2226
+ { "writeAll", file_writeAll, 0},
2227
+ { "list", file_list, 0},
2228
+ { "mkdir", file_mkdir, 0},
2229
+ { "toString", file_toString, 0},
2230
+ { "toURL", file_toURL, 0},
2231
+ {0}
2232
+ };
2233
+
2234
+ enum file_tinyid {
2235
+ FILE_LENGTH = -2,
2236
+ FILE_PARENT = -3,
2237
+ FILE_PATH = -4,
2238
+ FILE_NAME = -5,
2239
+ FILE_ISDIR = -6,
2240
+ FILE_ISFILE = -7,
2241
+ FILE_EXISTS = -8,
2242
+ FILE_CANREAD = -9,
2243
+ FILE_CANWRITE = -10,
2244
+ FILE_OPEN = -11,
2245
+ FILE_TYPE = -12,
2246
+ FILE_MODE = -13,
2247
+ FILE_CREATED = -14,
2248
+ FILE_MODIFIED = -15,
2249
+ FILE_SIZE = -16,
2250
+ FILE_RANDOMACCESS = -17,
2251
+ FILE_POSITION = -18,
2252
+ FILE_APPEND = -19,
2253
+ FILE_REPLACE = -20,
2254
+ FILE_AUTOFLUSH = -21,
2255
+ FILE_ISNATIVE = -22,
2256
+ };
2257
+
2258
+ static JSPropertySpec file_props[] = {
2259
+ {"length", FILE_LENGTH, JSPROP_ENUMERATE | JSPROP_READONLY },
2260
+ {"parent", FILE_PARENT, JSPROP_ENUMERATE | JSPROP_READONLY },
2261
+ {"path", FILE_PATH, JSPROP_ENUMERATE | JSPROP_READONLY },
2262
+ {"name", FILE_NAME, JSPROP_ENUMERATE | JSPROP_READONLY },
2263
+ {"isDirectory", FILE_ISDIR, JSPROP_ENUMERATE | JSPROP_READONLY },
2264
+ {"isFile", FILE_ISFILE, JSPROP_ENUMERATE | JSPROP_READONLY },
2265
+ {"exists", FILE_EXISTS, JSPROP_ENUMERATE | JSPROP_READONLY },
2266
+ {"canRead", FILE_CANREAD, JSPROP_ENUMERATE | JSPROP_READONLY },
2267
+ {"canWrite", FILE_CANWRITE, JSPROP_ENUMERATE | JSPROP_READONLY },
2268
+ {"canAppend", FILE_APPEND, JSPROP_ENUMERATE | JSPROP_READONLY },
2269
+ {"canReplace", FILE_REPLACE, JSPROP_ENUMERATE | JSPROP_READONLY },
2270
+ {"isOpen", FILE_OPEN, JSPROP_ENUMERATE | JSPROP_READONLY },
2271
+ {"type", FILE_TYPE, JSPROP_ENUMERATE | JSPROP_READONLY },
2272
+ {"mode", FILE_MODE, JSPROP_ENUMERATE | JSPROP_READONLY },
2273
+ {"creationTime", FILE_CREATED, JSPROP_ENUMERATE | JSPROP_READONLY },
2274
+ {"lastModified", FILE_MODIFIED, JSPROP_ENUMERATE | JSPROP_READONLY },
2275
+ {"size", FILE_SIZE, JSPROP_ENUMERATE | JSPROP_READONLY },
2276
+ {"hasRandomAccess", FILE_RANDOMACCESS, JSPROP_ENUMERATE | JSPROP_READONLY },
2277
+ {"hasAutoFlush", FILE_AUTOFLUSH, JSPROP_ENUMERATE | JSPROP_READONLY },
2278
+ {"position", FILE_POSITION, JSPROP_ENUMERATE },
2279
+ {"isNative", FILE_ISNATIVE, JSPROP_ENUMERATE | JSPROP_READONLY },
2280
+ {0}
2281
+ };
2282
+
2283
+ /* ------------------------- Property getter/setter ------------------------- */
2284
+ static JSBool
2285
+ file_getProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
2286
+ {
2287
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
2288
+ char *bytes;
2289
+ JSString *str;
2290
+ jsint tiny;
2291
+ PRFileInfo info;
2292
+ JSBool flag;
2293
+ PRExplodedTime expandedTime;
2294
+
2295
+ tiny = JSVAL_TO_INT(id);
2296
+ if (!file)
2297
+ return JS_TRUE;
2298
+
2299
+ switch (tiny) {
2300
+ case FILE_PARENT:
2301
+ SECURITY_CHECK(cx, NULL, "parent", file);
2302
+ if (!js_parent(cx, file, vp))
2303
+ return JS_FALSE;
2304
+ break;
2305
+ case FILE_PATH:
2306
+ str = JS_NewStringCopyZ(cx, file->path);
2307
+ if (!str)
2308
+ return JS_FALSE;
2309
+ *vp = STRING_TO_JSVAL(str);
2310
+ break;
2311
+ case FILE_NAME:
2312
+ if (!js_name(cx, file, vp))
2313
+ return JS_FALSE;
2314
+ break;
2315
+ case FILE_ISDIR:
2316
+ SECURITY_CHECK(cx, NULL, "isDirectory", file);
2317
+ *vp = BOOLEAN_TO_JSVAL(js_isDirectory(cx, file));
2318
+ break;
2319
+ case FILE_ISFILE:
2320
+ SECURITY_CHECK(cx, NULL, "isFile", file);
2321
+ *vp = BOOLEAN_TO_JSVAL(js_isFile(cx, file));
2322
+ break;
2323
+ case FILE_EXISTS:
2324
+ SECURITY_CHECK(cx, NULL, "exists", file);
2325
+ *vp = BOOLEAN_TO_JSVAL(js_exists(cx, file));
2326
+ break;
2327
+ case FILE_ISNATIVE:
2328
+ SECURITY_CHECK(cx, NULL, "isNative", file);
2329
+ *vp = BOOLEAN_TO_JSVAL(file->isNative);
2330
+ break;
2331
+ case FILE_CANREAD:
2332
+ SECURITY_CHECK(cx, NULL, "canRead", file);
2333
+ *vp = BOOLEAN_TO_JSVAL(js_canRead(cx, file));
2334
+ break;
2335
+ case FILE_CANWRITE:
2336
+ SECURITY_CHECK(cx, NULL, "canWrite", file);
2337
+ *vp = BOOLEAN_TO_JSVAL(js_canWrite(cx, file));
2338
+ break;
2339
+ case FILE_OPEN:
2340
+ SECURITY_CHECK(cx, NULL, "isOpen", file);
2341
+ *vp = BOOLEAN_TO_JSVAL(file->isOpen);
2342
+ break;
2343
+ case FILE_APPEND :
2344
+ SECURITY_CHECK(cx, NULL, "canAppend", file);
2345
+ JSFILE_CHECK_OPEN("canAppend");
2346
+ *vp = BOOLEAN_TO_JSVAL(!file->isNative &&
2347
+ (file->mode&PR_APPEND)==PR_APPEND);
2348
+ break;
2349
+ case FILE_REPLACE :
2350
+ SECURITY_CHECK(cx, NULL, "canReplace", file);
2351
+ JSFILE_CHECK_OPEN("canReplace");
2352
+ *vp = BOOLEAN_TO_JSVAL(!file->isNative &&
2353
+ (file->mode&PR_TRUNCATE)==PR_TRUNCATE);
2354
+ break;
2355
+ case FILE_AUTOFLUSH :
2356
+ SECURITY_CHECK(cx, NULL, "hasAutoFlush", file);
2357
+ JSFILE_CHECK_OPEN("hasAutoFlush");
2358
+ *vp = BOOLEAN_TO_JSVAL(!file->isNative && file->hasAutoflush);
2359
+ break;
2360
+ case FILE_TYPE:
2361
+ SECURITY_CHECK(cx, NULL, "type", file);
2362
+ JSFILE_CHECK_OPEN("type");
2363
+ if(js_isDirectory(cx, file)){
2364
+ *vp = JSVAL_VOID;
2365
+ break;
2366
+ }
2367
+
2368
+ switch (file->type) {
2369
+ case ASCII:
2370
+ *vp = STRING_TO_JSVAL(JS_NewStringCopyZ(cx, asciistring));
2371
+ break;
2372
+ case UTF8:
2373
+ *vp = STRING_TO_JSVAL(JS_NewStringCopyZ(cx, utfstring));
2374
+ break;
2375
+ case UCS2:
2376
+ *vp = STRING_TO_JSVAL(JS_NewStringCopyZ(cx, unicodestring));
2377
+ break;
2378
+ default:
2379
+ JS_ReportWarning(cx, "Unsupported file type %d, proceeding",
2380
+ file->type);
2381
+ }
2382
+ break;
2383
+ case FILE_MODE:
2384
+ SECURITY_CHECK(cx, NULL, "mode", file);
2385
+ JSFILE_CHECK_OPEN("mode");
2386
+ bytes = JS_malloc(cx, MODE_SIZE);
2387
+ bytes[0] = '\0';
2388
+ flag = JS_FALSE;
2389
+
2390
+ if ((file->mode&PR_RDONLY)==PR_RDONLY) {
2391
+ if (flag) strcat(bytes, ",");
2392
+ strcat(bytes, "read");
2393
+ flag = JS_TRUE;
2394
+ }
2395
+ if ((file->mode&PR_WRONLY)==PR_WRONLY) {
2396
+ if (flag) strcat(bytes, ",");
2397
+ strcat(bytes, "write");
2398
+ flag = JS_TRUE;
2399
+ }
2400
+ if ((file->mode&PR_RDWR)==PR_RDWR) {
2401
+ if (flag) strcat(bytes, ",");
2402
+ strcat(bytes, "readWrite");
2403
+ flag = JS_TRUE;
2404
+ }
2405
+ if ((file->mode&PR_APPEND)==PR_APPEND) {
2406
+ if (flag) strcat(bytes, ",");
2407
+ strcat(bytes, "append");
2408
+ flag = JS_TRUE;
2409
+ }
2410
+ if ((file->mode&PR_CREATE_FILE)==PR_CREATE_FILE) {
2411
+ if (flag) strcat(bytes, ",");
2412
+ strcat(bytes, "create");
2413
+ flag = JS_TRUE;
2414
+ }
2415
+ if ((file->mode&PR_TRUNCATE)==PR_TRUNCATE) {
2416
+ if (flag) strcat(bytes, ",");
2417
+ strcat(bytes, "replace");
2418
+ flag = JS_TRUE;
2419
+ }
2420
+ if (file->hasAutoflush) {
2421
+ if (flag) strcat(bytes, ",");
2422
+ strcat(bytes, "hasAutoFlush");
2423
+ flag = JS_TRUE;
2424
+ }
2425
+ *vp = STRING_TO_JSVAL(JS_NewStringCopyZ(cx, bytes));
2426
+ JS_free(cx, bytes);
2427
+ break;
2428
+ case FILE_CREATED:
2429
+ SECURITY_CHECK(cx, NULL, "creationTime", file);
2430
+ JSFILE_CHECK_NATIVE("creationTime");
2431
+ if(((file->isOpen)?
2432
+ PR_GetOpenFileInfo(file->handle, &info):
2433
+ PR_GetFileInfo(file->path, &info))!=PR_SUCCESS){
2434
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2435
+ JSFILEMSG_CANNOT_ACCESS_FILE_STATUS, file->path);
2436
+ goto out;
2437
+ }
2438
+
2439
+ PR_ExplodeTime(info.creationTime, PR_LocalTimeParameters,&expandedTime);
2440
+ *vp = OBJECT_TO_JSVAL(js_NewDateObject(cx, expandedTime.tm_year,
2441
+ expandedTime.tm_month,
2442
+ expandedTime.tm_mday,
2443
+ expandedTime.tm_hour,
2444
+ expandedTime.tm_min,
2445
+ expandedTime.tm_sec));
2446
+ break;
2447
+ case FILE_MODIFIED:
2448
+ SECURITY_CHECK(cx, NULL, "lastModified", file);
2449
+ JSFILE_CHECK_NATIVE("lastModified");
2450
+ if(((file->isOpen)?
2451
+ PR_GetOpenFileInfo(file->handle, &info):
2452
+ PR_GetFileInfo(file->path, &info))!=PR_SUCCESS){
2453
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2454
+ JSFILEMSG_CANNOT_ACCESS_FILE_STATUS, file->path);
2455
+ goto out;
2456
+ }
2457
+
2458
+ PR_ExplodeTime(info.modifyTime, PR_LocalTimeParameters, &expandedTime);
2459
+ *vp = OBJECT_TO_JSVAL(js_NewDateObject(cx, expandedTime.tm_year,
2460
+ expandedTime.tm_month,
2461
+ expandedTime.tm_mday,
2462
+ expandedTime.tm_hour,
2463
+ expandedTime.tm_min,
2464
+ expandedTime.tm_sec));
2465
+ break;
2466
+ case FILE_SIZE:
2467
+ SECURITY_CHECK(cx, NULL, "size", file);
2468
+ *vp = js_size(cx, file);
2469
+ break;
2470
+ case FILE_LENGTH:
2471
+ SECURITY_CHECK(cx, NULL, "length", file);
2472
+ JSFILE_CHECK_NATIVE("length");
2473
+
2474
+ if (js_isDirectory(cx, file)) { /* XXX debug me */
2475
+ PRDir *dir;
2476
+ PRDirEntry *entry;
2477
+ jsint count = 0;
2478
+
2479
+ if(!(dir = PR_OpenDir(file->path))){
2480
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2481
+ JSFILEMSG_CANNOT_OPEN_DIR, file->path);
2482
+ goto out;
2483
+ }
2484
+
2485
+ while ((entry = PR_ReadDir(dir, PR_SKIP_BOTH))) {
2486
+ count++;
2487
+ }
2488
+
2489
+ if(!PR_CloseDir(dir)){
2490
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2491
+ JSFILEMSG_OP_FAILED, "close", file->path);
2492
+
2493
+ goto out;
2494
+ }
2495
+
2496
+ *vp = INT_TO_JSVAL(count);
2497
+ break;
2498
+ }else{
2499
+ /* return file size */
2500
+ *vp = js_size(cx, file);
2501
+ }
2502
+ break;
2503
+ case FILE_RANDOMACCESS:
2504
+ SECURITY_CHECK(cx, NULL, "hasRandomAccess", file);
2505
+ JSFILE_CHECK_OPEN("hasRandomAccess");
2506
+ *vp = BOOLEAN_TO_JSVAL(file->hasRandomAccess);
2507
+ break;
2508
+ case FILE_POSITION:
2509
+ SECURITY_CHECK(cx, NULL, "position", file);
2510
+ JSFILE_CHECK_NATIVE("position");
2511
+ JSFILE_CHECK_OPEN("position");
2512
+
2513
+ if(!file->hasRandomAccess){
2514
+ JS_ReportWarning(cx, "File %s doesn't support random access, can't report the position, proceeding");
2515
+ *vp = JSVAL_VOID;
2516
+ break;
2517
+ }
2518
+
2519
+ if (file->isOpen && js_isFile(cx, file)) {
2520
+ int pos = PR_Seek(file->handle, 0, PR_SEEK_CUR);
2521
+ if(pos!=-1){
2522
+ *vp = INT_TO_JSVAL(pos);
2523
+ }else{
2524
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2525
+ JSFILEMSG_CANNOT_REPORT_POSITION, file->path);
2526
+ goto out;
2527
+ }
2528
+ }else {
2529
+ JS_ReportWarning(cx, "File %s is closed or not a plain file,"
2530
+ " can't report position, proceeding");
2531
+ goto out;
2532
+ }
2533
+ break;
2534
+ default:
2535
+ SECURITY_CHECK(cx, NULL, "file_access", file);
2536
+
2537
+ /* this is some other property -- try to use the dir["file"] syntax */
2538
+ if (js_isDirectory(cx, file)) {
2539
+ PRDir *dir = NULL;
2540
+ PRDirEntry *entry = NULL;
2541
+ char *prop_name;
2542
+
2543
+ str = JS_ValueToString(cx, id);
2544
+ if (!str)
2545
+ return JS_FALSE;
2546
+
2547
+ prop_name = JS_GetStringBytes(str);
2548
+
2549
+ /* no native files past this point */
2550
+ dir = PR_OpenDir(file->path);
2551
+ if(!dir) {
2552
+ /* This is probably not a directory */
2553
+ JS_ReportWarning(cx, "Can't open directory %s", file->path);
2554
+ return JS_FALSE;
2555
+ }
2556
+
2557
+ while ((entry = PR_ReadDir(dir, PR_SKIP_NONE)) != NULL) {
2558
+ if (!strcmp(entry->name, prop_name)){
2559
+ bytes = js_combinePath(cx, file->path, prop_name);
2560
+ *vp = OBJECT_TO_JSVAL(js_NewFileObject(cx, bytes));
2561
+ PR_CloseDir(dir);
2562
+ JS_free(cx, bytes);
2563
+ return !JSVAL_IS_NULL(*vp);
2564
+ }
2565
+ }
2566
+ PR_CloseDir(dir);
2567
+ }
2568
+ }
2569
+ return JS_TRUE;
2570
+
2571
+ out:
2572
+ return JS_FALSE;
2573
+ }
2574
+
2575
+ static JSBool
2576
+ file_setProperty(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
2577
+ {
2578
+ JSFile *file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
2579
+ jsint slot;
2580
+
2581
+ if (JSVAL_IS_STRING(id)){
2582
+ return JS_TRUE;
2583
+ }
2584
+
2585
+ slot = JSVAL_TO_INT(id);
2586
+
2587
+ switch (slot) {
2588
+ /* File.position = 10 */
2589
+ case FILE_POSITION:
2590
+ SECURITY_CHECK(cx, NULL, "set_position", file);
2591
+ JSFILE_CHECK_NATIVE("set_position");
2592
+
2593
+ if(!file->hasRandomAccess){
2594
+ JS_ReportWarning(cx, "File %s doesn't support random access, can't "
2595
+ "report the position, proceeding");
2596
+ goto out;
2597
+ }
2598
+
2599
+ if (file->isOpen && js_isFile(cx, file)) {
2600
+ int32 pos;
2601
+ int32 offset;
2602
+
2603
+ if (!JS_ValueToInt32(cx, *vp, &offset)){
2604
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2605
+ JSFILEMSG_FIRST_ARGUMENT_MUST_BE_A_NUMBER, "position", *vp);
2606
+ goto out;
2607
+ }
2608
+
2609
+ pos = PR_Seek(file->handle, offset, PR_SEEK_SET);
2610
+
2611
+ if(pos!=-1){
2612
+ *vp = INT_TO_JSVAL(pos);
2613
+ }else{
2614
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2615
+ JSFILEMSG_CANNOT_SET_POSITION, file->path);
2616
+ goto out;
2617
+ }
2618
+ } else {
2619
+ JS_ReportWarning(cx, "File %s is closed or not a file, can't set "
2620
+ "position, proceeding", file->path);
2621
+ goto out;
2622
+ }
2623
+ }
2624
+
2625
+ return JS_TRUE;
2626
+ out:
2627
+ return JS_FALSE;
2628
+ }
2629
+
2630
+ /*
2631
+ File.currentDir = new File("D:\") or File.currentDir = "D:\"
2632
+ */
2633
+ static JSBool
2634
+ file_currentDirSetter(JSContext *cx, JSObject *obj, jsval id, jsval *vp)
2635
+ {
2636
+ JSFile *file;
2637
+
2638
+ file = JS_GetInstancePrivate(cx, obj, &js_FileClass, NULL);
2639
+
2640
+ /* Look at the rhs and extract a file object from it */
2641
+ if (JSVAL_IS_OBJECT(*vp)) {
2642
+ if (JS_InstanceOf(cx, obj, &js_FileClass, NULL)) {
2643
+ /* Braindamaged rhs -- just return the old value */
2644
+ if (file && (!js_exists(cx, file) || !js_isDirectory(cx, file))) {
2645
+ JS_GetProperty(cx, obj, CURRENTDIR_PROPERTY, vp);
2646
+ return JS_FALSE;
2647
+ } else {
2648
+ chdir(file->path);
2649
+ return JS_TRUE;
2650
+ }
2651
+ } else {
2652
+ return JS_FALSE;
2653
+ }
2654
+ } else {
2655
+ JSObject *rhsObject;
2656
+ char *path;
2657
+
2658
+ path = JS_GetStringBytes(JS_ValueToString(cx, *vp));
2659
+ rhsObject = js_NewFileObject(cx, path);
2660
+ if (!rhsObject)
2661
+ return JS_FALSE;
2662
+
2663
+ if (!file || !js_exists(cx, file) || !js_isDirectory(cx, file)){
2664
+ JS_GetProperty(cx, obj, CURRENTDIR_PROPERTY, vp);
2665
+ } else {
2666
+ *vp = OBJECT_TO_JSVAL(rhsObject);
2667
+ chdir(path);
2668
+ }
2669
+ }
2670
+
2671
+ return JS_TRUE;
2672
+ }
2673
+
2674
+ /* Declare class */
2675
+ JSClass js_FileClass = {
2676
+ "File", JSCLASS_HAS_PRIVATE | JSCLASS_HAS_CACHED_PROTO(JSProto_File),
2677
+ JS_PropertyStub, JS_PropertyStub, file_getProperty, file_setProperty,
2678
+ JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, file_finalize
2679
+ };
2680
+
2681
+ /* -------------------- Functions exposed to the outside -------------------- */
2682
+ JS_PUBLIC_API(JSObject*)
2683
+ js_InitFileClass(JSContext *cx, JSObject* obj)
2684
+ {
2685
+ JSObject *file, *ctor, *afile;
2686
+ jsval vp;
2687
+ char *currentdir;
2688
+ char separator[2];
2689
+
2690
+ file = JS_InitClass(cx, obj, NULL, &js_FileClass, file_constructor, 1,
2691
+ file_props, file_functions, NULL, NULL);
2692
+ if (!file) {
2693
+ JS_ReportErrorNumber(cx, JSFile_GetErrorMessage, NULL,
2694
+ JSFILEMSG_INIT_FAILED);
2695
+ return NULL;
2696
+ }
2697
+
2698
+ ctor = JS_GetConstructor(cx, file);
2699
+ if (!ctor) return NULL;
2700
+
2701
+ /* Define CURRENTDIR property. We are doing this to get a
2702
+ slash at the end of the current dir */
2703
+ afile = js_NewFileObject(cx, CURRENT_DIR);
2704
+ currentdir = JS_malloc(cx, MAX_PATH_LENGTH);
2705
+ currentdir = getcwd(currentdir, MAX_PATH_LENGTH);
2706
+ afile = js_NewFileObject(cx, currentdir);
2707
+ JS_free(cx, currentdir);
2708
+ vp = OBJECT_TO_JSVAL(afile);
2709
+ JS_DefinePropertyWithTinyId(cx, ctor, CURRENTDIR_PROPERTY, 0, vp,
2710
+ JS_PropertyStub, file_currentDirSetter,
2711
+ JSPROP_ENUMERATE | JSPROP_READONLY );
2712
+
2713
+ /* Define input */
2714
+ vp = OBJECT_TO_JSVAL(js_NewFileObjectFromFILE(cx, stdin,
2715
+ STDINPUT_NAME, PR_RDONLY, JS_TRUE, JS_FALSE));
2716
+ JS_SetProperty(cx, ctor, "input", &vp);
2717
+
2718
+ /* Define output */
2719
+ vp = OBJECT_TO_JSVAL(js_NewFileObjectFromFILE(cx, stdout,
2720
+ STDOUTPUT_NAME, PR_WRONLY, JS_TRUE, JS_FALSE));
2721
+ JS_SetProperty(cx, ctor, "output", &vp);
2722
+
2723
+ /* Define error */
2724
+ vp = OBJECT_TO_JSVAL(js_NewFileObjectFromFILE(cx, stderr,
2725
+ STDERROR_NAME, PR_WRONLY, JS_TRUE, JS_FALSE));
2726
+ JS_SetProperty(cx, ctor, "error", &vp);
2727
+
2728
+ separator[0] = FILESEPARATOR;
2729
+ separator[1] = '\0';
2730
+ vp = STRING_TO_JSVAL(JS_NewStringCopyZ(cx, separator));
2731
+ JS_DefinePropertyWithTinyId(cx, ctor, SEPARATOR_PROPERTY, 0, vp,
2732
+ JS_PropertyStub, JS_PropertyStub,
2733
+ JSPROP_ENUMERATE | JSPROP_READONLY );
2734
+ return file;
2735
+ }
2736
+ #endif /* JS_HAS_FILE_OBJECT */