ResumeFileResultTests.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505
  1. using Masuit.Tools.Mvc;
  2. using Masuit.Tools.Mvc.ActionResults;
  3. using Masuit.Tools.UnitTest.Mvc.Mocks;
  4. using NUnit.Framework;
  5. using System;
  6. using System.IO;
  7. using System.Net;
  8. using System.Threading;
  9. namespace Masuit.Tools.UnitTest.Mvc
  10. {
  11. [TestFixture]
  12. public class ResumeFileResultTests : BaseTests
  13. {
  14. private FileInfo _file;
  15. private FileInfo _file2;
  16. [SetUp]
  17. public void Setup()
  18. {
  19. _file = new FileInfo(FilePath("download-test-file.txt"));
  20. _file2 = new FileInfo(FilePath("download-test-file2.txt"));
  21. Request.Headers.Clear();
  22. Response.ClearTestResponse();
  23. }
  24. [Test]
  25. public void CanCalculateEtagForFile()
  26. {
  27. Assert.IsNotNull(ResumeFileResult.Util.Etag(_file));
  28. }
  29. [Test]
  30. public void EtagDoesNotDependOnTime()
  31. {
  32. var etag1 = ResumeFileResult.Util.Etag(_file);
  33. Thread.Sleep(100);
  34. var etag2 = ResumeFileResult.Util.Etag(_file);
  35. Assert.AreEqual(etag1, etag2);
  36. }
  37. [Test]
  38. public void EtagDoesDependOnFile()
  39. {
  40. var etag1 = ResumeFileResult.Util.Etag(_file);
  41. Thread.Sleep(100);
  42. var etag2 = ResumeFileResult.Util.Etag(_file2);
  43. Assert.AreNotEqual(etag1, etag2);
  44. }
  45. [Test]
  46. public void IsNotModified_Is_False_If_IfNoneMatch_And_IfModifiedSince_Are_Empty()
  47. {
  48. Request.Headers[HttpHeaders.IfNoneMatch] = null;
  49. Request.Headers[HttpHeaders.IfModifiedSince] = null;
  50. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsNotModified());
  51. }
  52. [Test]
  53. public void IsNotModified_Is_False_If_Etag_Is_Invalid_And_IfModifiedSince_Is_Null()
  54. {
  55. var etag = "invalid etag";
  56. Request.Headers[HttpHeaders.IfNoneMatch] = etag;
  57. Request.Headers[HttpHeaders.IfModifiedSince] = null;
  58. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsNotModified());
  59. }
  60. [Test]
  61. public void IsNotModified_Is_True_If_Etag_Is_Valid()
  62. {
  63. var etag = ResumeFileResult.Util.Etag(_file);
  64. Request.Headers[HttpHeaders.IfNoneMatch] = etag;
  65. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).IsNotModified());
  66. }
  67. [Test]
  68. public void IsNotModified_Is_True_If_Etag_Is_Star()
  69. {
  70. var etag = "*";
  71. Request.Headers[HttpHeaders.IfNoneMatch] = etag;
  72. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).IsNotModified());
  73. }
  74. [Test]
  75. public void IsNotModified_Is_False_If_Etag_Is_Empty_And_IfModifiedSince_Is_Invalid()
  76. {
  77. Request.Headers[HttpHeaders.IfModifiedSince] = DateTime.Now.ToString("R");
  78. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsNotModified());
  79. }
  80. [Test]
  81. public void IsNotModified_Is_False_If_Etag_Is_Empty_And_IfModifiedSince_Is_LastFileWriteTime()
  82. {
  83. Request.Headers[HttpHeaders.IfModifiedSince] = _file.LastWriteTime.ToString("R");
  84. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).IsNotModified());
  85. }
  86. [Test]
  87. public void IsPreconditionFailedTest_Is_False_If_ifMatch_And_ifUnmodifiedSince_Are_Empty()
  88. {
  89. Request.Headers[HttpHeaders.IfMatch] = null;
  90. Request.Headers[HttpHeaders.IfUnmodifiedSince] = null;
  91. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsPreconditionFailed());
  92. }
  93. [Test]
  94. public void IsPreconditionFailedTest_Is_IsTrue_If_ifMatch_Doesnot_Match_Etag_Of_The_File()
  95. {
  96. Request.Headers[HttpHeaders.IfMatch] = "incorrect";
  97. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).IsPreconditionFailed());
  98. }
  99. [Test]
  100. public void IsPreconditionFailedTest_Is_IsFalse_If_ifMatch_Matches_Etag_Of_The_File()
  101. {
  102. Request.Headers[HttpHeaders.IfMatch] = ResumeFileResult.Util.Etag(_file);
  103. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsPreconditionFailed());
  104. }
  105. [Test]
  106. public void IsPreconditionFailedTest_Is_IsFalse_If_ifMatch_Equals_Star()
  107. {
  108. Request.Headers[HttpHeaders.IfMatch] = "*";
  109. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsPreconditionFailed());
  110. }
  111. [Test]
  112. public void IsPreconditionFailedTest_Is_IsTrue_If_ifUnmodifiedSince_Doesnot_Equal_FileLastWriteTime()
  113. {
  114. Request.Headers[HttpHeaders.IfUnmodifiedSince] = DateTime.Now.ToString("R");
  115. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).IsPreconditionFailed());
  116. }
  117. [Test]
  118. public void IsPreconditionFailedTest_Is_IsFalse_If_ifUnmodifiedSince_Equals_FileLastWriteTime()
  119. {
  120. Request.Headers[HttpHeaders.IfUnmodifiedSince] = _file.LastWriteTime.ToString("R");
  121. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsPreconditionFailed());
  122. }
  123. [Test]
  124. public void IsRangeNotSatisfiable_Is_True_If_Range_Header_Has_Invalid_Format()
  125. {
  126. Request.Headers[HttpHeaders.Range] = "blah";
  127. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).IsRangeNotSatisfiable());
  128. }
  129. [Test]
  130. public void IsRangeNotSatisfiable_Is_True_If_Start_Greater_Than_End()
  131. {
  132. Request.Headers[HttpHeaders.Range] = "bytes=100-0";
  133. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).IsRangeNotSatisfiable());
  134. }
  135. [Test]
  136. public void IsRangeNotSatisfiable_Is_True_If_End_Equals_Total_File_Size()
  137. {
  138. Request.Headers[HttpHeaders.Range] = "bytes=0-" + _file.Length;
  139. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).IsRangeNotSatisfiable());
  140. }
  141. [Test]
  142. public void IsRangeNotSatisfiable_Is_True_If_End_Greater_Than_Total_File_Size()
  143. {
  144. Request.Headers[HttpHeaders.Range] = "bytes=0-" + _file.Length + 10;
  145. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).IsRangeNotSatisfiable());
  146. }
  147. [Test]
  148. public void IsRangeNotSatisfiable_Is_False_If_Range_Header_Is_Null()
  149. {
  150. Request.Headers[HttpHeaders.Range] = null;
  151. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsRangeNotSatisfiable());
  152. }
  153. [Test]
  154. public void IsRangeNotSatisfiable_Is_False_If_Range_Has_StartsWith_Format()
  155. {
  156. Request.Headers[HttpHeaders.Range] = "bytes=0-";
  157. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsRangeNotSatisfiable());
  158. }
  159. [Test]
  160. public void IsRangeNotSatisfiable_Is_False_If_Range_Has_LastXbytes_Format()
  161. {
  162. Request.Headers[HttpHeaders.Range] = "bytes=-100";
  163. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsRangeNotSatisfiable());
  164. }
  165. [Test]
  166. public void IsRangeNotSatisfiable_Is_False_If_Range_Ends_With_Last_Byte_Position()
  167. {
  168. Request.Headers[HttpHeaders.Range] = "bytes=100-" + (_file.Length - 1);
  169. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).IsRangeNotSatisfiable());
  170. }
  171. [Test]
  172. public void SendRange_Is_False_If_Range_And_ifRange_Headers_Are_Null()
  173. {
  174. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).SendRange());
  175. }
  176. [Test]
  177. public void SendRange_Is_False_If_Range_Is_Null_And_ifRange_Is_Correct()
  178. {
  179. Request.Headers[HttpHeaders.IfRange] = ResumeFileResult.Util.Etag(_file);
  180. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).SendRange());
  181. }
  182. [Test]
  183. public void SendRange_Is_True_If_Range_Is_Correct_And_ifRange_Is_Null()
  184. {
  185. Request.Headers[HttpHeaders.Range] = "bytes=0-100";
  186. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).SendRange());
  187. }
  188. [Test]
  189. public void SendRange_Is_True_If_Range_And_ifRange_Are_Correct()
  190. {
  191. Request.Headers[HttpHeaders.IfRange] = ResumeFileResult.Util.Etag(_file);
  192. Request.Headers[HttpHeaders.Range] = "bytes=0-100";
  193. Assert.IsTrue(new MockResumeFileResult(_file.FullName, Request).SendRange());
  194. }
  195. [Test]
  196. public void SendRange_Is_False_If_Range_Is_Correct_But_ifRange_Is_InCorrect()
  197. {
  198. Request.Headers[HttpHeaders.IfRange] = "incorrect etag";
  199. Request.Headers[HttpHeaders.Range] = "bytes=0-100";
  200. Assert.IsFalse(new MockResumeFileResult(_file.FullName, Request).SendRange());
  201. }
  202. [Test]
  203. public void HeadersTest_Should_Not_Send_File_If_File_Has_Not_Been_Changed()
  204. {
  205. Request.Headers[HttpHeaders.IfNoneMatch] = ResumeFileResult.Util.Etag(_file);
  206. var result = new MockResumeFileResult(_file.FullName, Request);
  207. Assert.IsTrue(result.IsNotModified());
  208. result.WriteFileTest(Response);
  209. Assert.AreEqual((int)HttpStatusCode.NotModified, Response.StatusCode);
  210. Assert.IsNotNull(Response.Headers[HttpHeaders.Etag]);
  211. Assert.IsNotNull(Response.Headers[HttpHeaders.Expires]);
  212. Assert.IsNotNull(Response.Headers[HttpHeaders.LastModified]);
  213. Assert.IsNull(Response.Headers[HttpHeaders.ContentRange]);
  214. Assert.AreEqual(0, Response.OutputStream.Length);
  215. }
  216. [Test]
  217. public void HeadersTest_Should_Not_Send_File_IfPreconditionFailed()
  218. {
  219. Request.Headers[HttpHeaders.IfMatch] = "invalid";
  220. var result = new MockResumeFileResult(_file.FullName, Request);
  221. Assert.IsTrue(result.IsPreconditionFailed());
  222. result.WriteFileTest(Response);
  223. Assert.AreEqual((int)HttpStatusCode.PreconditionFailed, Response.StatusCode);
  224. Assert.IsNotNull(Response.Headers[HttpHeaders.Etag]);
  225. Assert.IsNotNull(Response.Headers[HttpHeaders.Expires]);
  226. Assert.IsNotNull(Response.Headers[HttpHeaders.LastModified]);
  227. Assert.IsNull(Response.Headers[HttpHeaders.ContentRange]);
  228. Assert.AreEqual(0, Response.OutputStream.Length);
  229. }
  230. [Test]
  231. public void HeadersTest_Should_Not_Send_File_Is_RangeNotSatisfiable()
  232. {
  233. Request.Headers[HttpHeaders.Range] = "invalid";
  234. var result = new MockResumeFileResult(_file.FullName, Request);
  235. Assert.IsTrue(result.IsRangeNotSatisfiable());
  236. result.WriteFileTest(Response);
  237. Assert.AreEqual((int)HttpStatusCode.RequestedRangeNotSatisfiable, Response.StatusCode);
  238. Assert.IsNotNull(Response.Headers[HttpHeaders.Etag]);
  239. Assert.IsNotNull(Response.Headers[HttpHeaders.Expires]);
  240. Assert.IsNotNull(Response.Headers[HttpHeaders.LastModified]);
  241. Assert.AreEqual("bytes */" + _file.Length, Response.Headers[HttpHeaders.ContentRange]);
  242. Assert.AreEqual(0, Response.OutputStream.Length);
  243. }
  244. [Test]
  245. public void HeadersTest_Should_Send_File_If_All_Headers_Are_Null()
  246. {
  247. var result = new MockResumeFileResult(_file.FullName, Request);
  248. result.WriteFileTest(Response);
  249. Assert.AreEqual((int)HttpStatusCode.OK, Response.StatusCode);
  250. Assert.IsNotNull(Response.Headers[HttpHeaders.Etag]);
  251. Assert.IsNotNull(Response.Headers[HttpHeaders.Expires]);
  252. Assert.IsNotNull(Response.Headers[HttpHeaders.LastModified]);
  253. Assert.IsNotNull(Response.Headers[HttpHeaders.ContentRange]);
  254. Assert.IsNotNull(Response.Headers[HttpHeaders.ContentLength]);
  255. Assert.AreEqual(_file.Length, Response.OutputStream.Length);
  256. }
  257. [Test]
  258. public void Range_First_500b()
  259. {
  260. var stream = GetResponseStream("bytes=0-499");
  261. Assert.AreEqual(500, stream.Length);
  262. Assert.AreEqual(206, Response.StatusCode);
  263. Assert.AreEqual($"bytes 0-499/{_file.Length}", Response.Headers[HttpHeaders.ContentRange]);
  264. }
  265. [Test]
  266. public void Range_From_500b_to_899b()
  267. {
  268. var stream = GetResponseStream("bytes=500-899");
  269. Assert.AreEqual(400, stream.Length);
  270. Assert.AreEqual(206, Response.StatusCode);
  271. Assert.AreEqual($"bytes 500-899/{_file.Length}", Response.Headers[HttpHeaders.ContentRange]);
  272. }
  273. [Test]
  274. public void Range_Last_300b()
  275. {
  276. var stream = GetResponseStream("bytes=-300");
  277. Assert.AreEqual(300, stream.Length);
  278. Assert.AreEqual(206, Response.StatusCode);
  279. var from = _file.Length - 300;
  280. var to = _file.Length - 1;
  281. Assert.AreEqual($"bytes {from}-{to}/{_file.Length}", Response.Headers[HttpHeaders.ContentRange]);
  282. }
  283. [Test]
  284. public void Range_From_100b_toThe_End()
  285. {
  286. var stream = GetResponseStream($"bytes={(_file.Length - 100)}-");
  287. Assert.AreEqual(100, stream.Length);
  288. Assert.AreEqual(206, Response.StatusCode);
  289. var from = _file.Length - 100;
  290. var to = _file.Length - 1;
  291. Assert.AreEqual($"bytes {from}-{to}/{_file.Length}", Response.Headers[HttpHeaders.ContentRange]);
  292. }
  293. [Test]
  294. public void Range_First_1b()
  295. {
  296. var stream = GetResponseStream("bytes=0-0");
  297. Assert.AreEqual(1, stream.Length);
  298. Assert.AreEqual(206, Response.StatusCode);
  299. Assert.AreEqual($"bytes 0-0/{_file.Length}", Response.Headers[HttpHeaders.ContentRange]);
  300. }
  301. [Test]
  302. public void Range_Last_1b()
  303. {
  304. var stream = GetResponseStream("bytes=-1");
  305. Assert.AreEqual(1, stream.Length);
  306. Assert.AreEqual(206, Response.StatusCode);
  307. var from = _file.Length - 1;
  308. var to = _file.Length - 1;
  309. Assert.AreEqual($"bytes {from}-{to}/{_file.Length}", Response.Headers[HttpHeaders.ContentRange]);
  310. }
  311. [Test]
  312. public void Range_Whole_File_With_RangeHeader()
  313. {
  314. var stream = GetResponseStream("bytes=0-" + (_file.Length - 1));
  315. Assert.AreEqual(_file.Length, stream.Length);
  316. Assert.AreEqual(206, Response.StatusCode);
  317. Assert.AreEqual($"bytes 0-{(_file.Length - 1)}/{_file.Length}", Response.Headers[HttpHeaders.ContentRange]);
  318. }
  319. [Test]
  320. public void Range_Whole_File_Without_RangeHeader()
  321. {
  322. var stream = GetResponseStream(null);
  323. Assert.AreEqual(_file.Length, stream.Length);
  324. Assert.AreEqual(200, Response.StatusCode);
  325. Assert.AreEqual($"bytes 0-{(_file.Length - 1)}/{_file.Length}", Response.Headers[HttpHeaders.ContentRange]);
  326. }
  327. [Test]
  328. public void TransmissionRange_From_0_To_0()
  329. {
  330. Request.Headers[HttpHeaders.Range] = "bytes=0-0";
  331. new MockResumeFileResult(_file.FullName, Request).WriteFileTest(Response);
  332. Assert.AreEqual(1, Response.OutputStream.Length);
  333. AssertBytes(_file, Response.OutputStream, 0, 1);
  334. }
  335. [Test]
  336. public void TransmissionRange_From_1_To_100()
  337. {
  338. Request.Headers[HttpHeaders.Range] = "bytes=1-100";
  339. new MockResumeFileResult(_file.FullName, Request).WriteFileTest(Response);
  340. Assert.AreEqual(100, Response.OutputStream.Length);
  341. AssertBytes(_file, Response.OutputStream, 1, 100);
  342. }
  343. [Test]
  344. public void TransmissionRange_From_101_To_theEnd()
  345. {
  346. Request.Headers[HttpHeaders.Range] = "bytes=101-";
  347. new MockResumeFileResult(_file.FullName, Request).WriteFileTest(Response);
  348. Assert.AreEqual(_file.Length - 101, Response.OutputStream.Length);
  349. AssertBytes(_file, Response.OutputStream, 101, (int)_file.Length);
  350. }
  351. [Test]
  352. public void TransmissionRange_WholeFile_WithRangeHeader()
  353. {
  354. Request.Headers[HttpHeaders.Range] = "bytes=0-";
  355. new MockResumeFileResult(_file.FullName, Request).WriteFileTest(Response);
  356. Assert.AreEqual(_file.Length, Response.OutputStream.Length);
  357. AssertBytes(_file, Response.OutputStream, 0, (int)_file.Length);
  358. }
  359. [Test]
  360. public void TransmissionRange_WholeFile_WithoutRangeHeader()
  361. {
  362. Request.Headers[HttpHeaders.Range] = null;
  363. new MockResumeFileResult(_file.FullName, Request).WriteFileTest(Response);
  364. Assert.AreEqual(_file.Length, Response.OutputStream.Length);
  365. AssertBytes(_file, Response.OutputStream, 0, (int)_file.Length);
  366. }
  367. [Test]
  368. public void ShouldSend206If_Range_HeaderExists()
  369. {
  370. Request.Headers[HttpHeaders.Range] = "bytes=0-";
  371. new MockResumeFileResult(_file.FullName, Request).WriteFileTest(Response);
  372. Assert.AreEqual(206, Response.StatusCode);
  373. }
  374. [Test]
  375. public void ShouldSend200If_Range_HeaderDoesNotExist()
  376. {
  377. Request.Headers[HttpHeaders.Range] = null;
  378. new MockResumeFileResult(_file.FullName, Request).WriteFileTest(Response);
  379. Assert.AreEqual(200, Response.StatusCode);
  380. }
  381. [Test]
  382. public void IfRangeHeader_Should_Be_Ignored_If_ItNotEquals_Etag()
  383. {
  384. Request.Headers[HttpHeaders.IfRange] = "ifRange fake header";
  385. var mock = new MockResumeFileResult(_file.FullName, Request);
  386. mock.WriteFileTest(Response);
  387. Assert.AreNotEqual(ResumeFileResult.Util.Etag(_file), Request.Headers[HttpHeaders.IfRange]);
  388. Assert.AreEqual(200, Response.StatusCode);
  389. }
  390. [Test]
  391. public void Etag_Should_Be_Added_To_Response_If_It_Equals_With_IfRange_In_Request()
  392. {
  393. var etag = ResumeFileResult.Util.Etag(_file);
  394. Request.Headers[HttpHeaders.IfRange] = etag;
  395. var mock = new MockResumeFileResult(_file.FullName, Request);
  396. mock.WriteFileTest(Response);
  397. Assert.AreEqual(Response.Headers[HttpHeaders.Etag], etag);
  398. Assert.AreEqual(200, Response.StatusCode);
  399. }
  400. [Test]
  401. public void Etag_Should_Be_Added_To_Response_If_It_Equals_With_IfRange_In_Request__PartialResponse()
  402. {
  403. var etag = ResumeFileResult.Util.Etag(_file);
  404. Request.Headers[HttpHeaders.IfRange] = etag;
  405. Request.Headers[HttpHeaders.Range] = "bytes=0-";
  406. new MockResumeFileResult(_file.FullName, Request).WriteFileTest(Response);
  407. Assert.AreEqual(Response.Headers[HttpHeaders.Etag], etag);
  408. Assert.IsNotNull(Response.Headers[HttpHeaders.ContentRange]);
  409. Assert.AreEqual(206, Response.StatusCode);
  410. }
  411. [Test]
  412. public void It_Should_Attach_Content_Disposition_If_There_Is_Download_File_Name()
  413. {
  414. new MockResumeFileResult(_file.FullName, Request, "test.name").WriteFileTest(Response);
  415. Assert.IsNotNull(Response.Headers[HttpHeaders.ContentDisposition]);
  416. }
  417. private Stream GetResponseStream(string range)
  418. {
  419. Response.ClearTestResponse();
  420. Response.StatusCode = 500;
  421. Request.Headers[HttpHeaders.Range] = range;
  422. new MockResumeFileResult(_file.FullName, Request).WriteFileTest(Response);
  423. return Response.OutputStream;
  424. }
  425. private void AssertBytes(FileInfo file, Stream responseStream, int from, int to)
  426. {
  427. using (var fileStream = file.OpenRead())
  428. {
  429. responseStream.Seek(0, SeekOrigin.Begin);
  430. fileStream.Seek(from, SeekOrigin.Begin);
  431. for (var byteIndex = from; byteIndex < to; byteIndex++)
  432. {
  433. var responseByte = responseStream.ReadByte();
  434. var fileByte = fileStream.ReadByte();
  435. Assert.AreEqual(responseByte, fileByte);
  436. }
  437. }
  438. }
  439. }
  440. }