/central/src/test/java/com/github/rinde/rinsim/central/arrays/ArraysSolverValidatorTest.java

http://github.com/rinde/RinSim · Java · 845 lines · 588 code · 103 blank · 154 comment · 0 complexity · d8fa1d42214e80c0b9c789be34407334 MD5 · raw file

  1. /*
  2. * Copyright (C) 2011-2018 Rinde R.S. van Lon
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package com.github.rinde.rinsim.central.arrays;
  17. import static com.github.rinde.rinsim.central.arrays.ArraysSolverValidator.validateInputs;
  18. import static com.github.rinde.rinsim.central.arrays.ArraysSolverValidator.validateOutputs;
  19. import static com.github.rinde.rinsim.central.arrays.ArraysSolverValidator.wrap;
  20. import static com.google.common.base.Preconditions.checkArgument;
  21. import java.util.Arrays;
  22. import javax.annotation.Nullable;
  23. import org.junit.Test;
  24. import com.github.rinde.rinsim.testutil.TestUtil;
  25. /**
  26. * You could see this class as the validator of the validator, so now you might
  27. * be thinking, who validates the validator of the validators? Well, that's YOU!
  28. * @author Rinde van Lon
  29. */
  30. public class ArraysSolverValidatorTest {
  31. /*
  32. * VALIDATE SINGLE VEHICLE INPUTS
  33. */
  34. @Test(expected = IllegalArgumentException.class)
  35. public void validateInputsEmptyTravelTimeMatrix() {
  36. validateInputs(new int[][] {}, new int[] {}, new int[] {}, new int[][] {},
  37. new int[] {});
  38. }
  39. @Test(expected = IllegalArgumentException.class)
  40. public void validateInputsInvalidTravelTimeMatrix1() {
  41. validateInputs(new int[7][6], new int[] {}, new int[] {}, new int[][] {},
  42. new int[] {});
  43. }
  44. @Test(expected = IllegalArgumentException.class)
  45. public void validateInputsInvalidTravelTimeMatrix2() {
  46. validateInputs(new int[][] {new int[3], new int[3], new int[4]},
  47. new int[] {}, new int[] {}, new int[][] {}, new int[] {});
  48. }
  49. @Test(expected = IllegalArgumentException.class)
  50. public void validateInputsInvalidReleaseDatesLength() {
  51. validateInputs(new int[4][4], new int[5], new int[] {}, new int[][] {},
  52. new int[] {});
  53. }
  54. @Test(expected = IllegalArgumentException.class)
  55. public void validateInputsInvalidDueDatesLength() {
  56. validateInputs(new int[4][4], new int[4], new int[2], new int[][] {},
  57. new int[] {});
  58. }
  59. @Test(expected = IllegalArgumentException.class)
  60. public void validateInputsInvalidServiceTime() {
  61. validateInputs(new int[7][7], new int[7], new int[7], new int[][] {},
  62. new int[3]);
  63. }
  64. @Test(expected = IllegalArgumentException.class)
  65. public void validateInputsInvalidTW() {
  66. validateInputs(new int[3][3], new int[] {1, 2, 3}, new int[] {1, 3, 2},
  67. new int[][] {}, new int[3]);
  68. }
  69. @Test(expected = IllegalArgumentException.class)
  70. public void validateInputsInvalidTWstart1() {
  71. validateInputs(new int[3][3], new int[] {1, 2, 3}, new int[] {1, 3, 6},
  72. new int[][] {}, new int[3]);
  73. }
  74. @Test(expected = IllegalArgumentException.class)
  75. public void validateInputsInvalidTWstart2() {
  76. validateInputs(new int[3][3], new int[] {0, 2, 3}, new int[] {1, 3, 6},
  77. new int[][] {}, new int[3]);
  78. }
  79. @Test
  80. public void validateInputsValidEmpty() {
  81. validateInputs(new int[3][3], new int[3], new int[3], new int[][] {},
  82. new int[3]);
  83. }
  84. @Test(expected = IllegalArgumentException.class)
  85. public void validateInputsInvalidPairSize() {
  86. validateInputs(new int[4][4], new int[4], new int[4], new int[][] {
  87. new int[] {1, 2}, new int[3]}, new int[4]);
  88. }
  89. @Test(expected = IllegalArgumentException.class)
  90. public void validateInputsInvalidPairLocation1() {
  91. validateInputs(new int[6][6], new int[6], new int[6], new int[][] {
  92. new int[] {1, 2}, new int[] {3, 0}}, new int[6]);
  93. }
  94. @Test(expected = IllegalArgumentException.class)
  95. public void validateInputsInvalidPairLocation2() {
  96. validateInputs(new int[6][6], new int[6], new int[6], new int[][] {
  97. new int[] {1, 2}, new int[] {3, 913}}, new int[6]);
  98. }
  99. @Test(expected = IllegalArgumentException.class)
  100. public void validateInputsDuplicatePairLocation() {
  101. validateInputs(new int[6][6], new int[6], new int[6], new int[][] {
  102. new int[] {1, 2}, new int[] {2, 0}}, new int[6]);
  103. }
  104. @Test(expected = IllegalArgumentException.class)
  105. public void validateInputsInvalidArrivalTimes() {
  106. validateInputs(new int[6][6], new int[6], new int[6], new int[][] {
  107. new int[] {4, 2}, new int[] {3, 1}}, new int[] {-1, -1, 0, 0, 0,
  108. 0});
  109. }
  110. @Test
  111. public void validateInputsValid() {
  112. validateInputs(new int[6][6], new int[6], new int[6], new int[][] {
  113. new int[] {4, 2}, new int[] {3, 1}}, new int[6]);
  114. }
  115. /*
  116. * VALIDATE MULTI VEHICLE INPUTS
  117. */
  118. @Test(expected = IllegalArgumentException.class)
  119. public void validateInputsInvalidNoVehicles() {
  120. validateInputs(new int[6][6], new int[6], new int[6], new int[][] {
  121. new int[] {4, 2}, new int[] {3, 1}}, new int[6], //
  122. new int[0][0], new int[0][0], new int[0], new int[0], null);
  123. }
  124. /**
  125. * Invalid array size.
  126. */
  127. @Test(expected = IllegalArgumentException.class)
  128. public void validateInputsInvalidVehicleTravelTimes1() {
  129. validateInputs(new int[6][6], new int[6], new int[6],
  130. // service pairs
  131. new int[][] {new int[] {4, 2}, new int[] {3, 1}},
  132. // service times
  133. new int[6],
  134. // vehicle travel times with invalid dimensions
  135. new int[2][0],
  136. //
  137. new int[0][0], new int[2], new int[2], null);
  138. }
  139. /**
  140. * Negative travel time.
  141. */
  142. @Test(expected = IllegalArgumentException.class)
  143. public void validateInputsInvalidVehicleTravelTimes2() {
  144. validateInputs(new int[6][6], new int[6], new int[6], new int[][] {
  145. new int[] {4, 2}, new int[] {3, 1}}, new int[6], //
  146. // vehicle travel times, with one invalid travel time (-1)
  147. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, -1, 0}}, //
  148. new int[0][0], new int[2], new int[2], null);
  149. }
  150. /**
  151. * Invalid vehicle travel time to self.
  152. */
  153. @Test(expected = IllegalArgumentException.class)
  154. public void validateInputsInvalidVehicleTravelTimes3() {
  155. validateInputs(new int[6][6], new int[6], new int[6],
  156. // service pairs
  157. new int[][] {new int[] {4, 2}, new int[] {3, 1}},
  158. // service times
  159. new int[6],
  160. // vehicle travel times with invalid travel time from second vehicle to
  161. // itself (must be 0)
  162. new int[][] {{0, 0, 0, 0, 0, 0}, {1, 0, 0, 0, 0, 0}},
  163. //
  164. new int[0][0], new int[2], new int[2], null);
  165. }
  166. /**
  167. * Travel time must be max int.
  168. */
  169. @Test(expected = IllegalArgumentException.class)
  170. public void validateInputsInvalidVehicleTravelTimes4() {
  171. validateInputs(new int[6][6], new int[6], new int[6],
  172. // service pairs
  173. new int[][] {new int[] {4, 2}, new int[] {3, 1}},
  174. // service times
  175. new int[6],
  176. // vehicle travel times with invalid travel time from first vehicle to
  177. // index 3 (value = 0, should be MI)
  178. new int[][] {{0, MI, MI - 1, 0, MI, 0}, {0, 0, 0, 0, 0, 0}},
  179. // inventories
  180. new int[0][0],
  181. // remaining service times
  182. new int[2],
  183. // current destinations
  184. new int[] {2, 0},
  185. // current solutions
  186. null);
  187. }
  188. /**
  189. * Normal travel time expected but is MI.
  190. */
  191. @Test(expected = IllegalArgumentException.class)
  192. public void validateInputsInvalidVehicleTravelTimes5() {
  193. validateInputs(new int[6][6], new int[6], new int[6],
  194. // service pairs
  195. new int[][] {new int[] {4, 2}, new int[] {3, 1}},
  196. // service times
  197. new int[6],
  198. // vehicle travel times with invalid travel time from first vehicle to
  199. // index 4 (value = MI, should be < MI)
  200. new int[][] {{0, 0, MI - 1, 0, MI, 0}, {0, 0, 0, 0, 0, 0}},
  201. // inventories
  202. new int[0][0],
  203. // remaining service times
  204. new int[2],
  205. // current destinations
  206. new int[] {0, 0},
  207. // current solutions
  208. null);
  209. }
  210. @Test(expected = IllegalArgumentException.class)
  211. public void validateInputsInvalidInventories1() {
  212. // inventory too big
  213. validateInputs(new int[6][6], new int[6], new int[6], new int[][] {
  214. new int[] {4, 2}, new int[] {3, 1}}, new int[6], //
  215. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  216. new int[][] {{0}}, new int[2], new int[2], null);
  217. }
  218. @Test(expected = IllegalArgumentException.class)
  219. public void validateInputsInvalidInventories2() {
  220. // inventory too small
  221. validateInputs(new int[6][6], new int[6], new int[6],
  222. new int[][] {new int[] {3, 1}}, new int[6], //
  223. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  224. new int[][] {{0}}, new int[2], new int[2], null);
  225. }
  226. @Test(expected = IllegalArgumentException.class)
  227. public void validateInputsInvalidInventories3() {
  228. // invalid inventory dimensions
  229. validateInputs(new int[6][6], new int[6], new int[6],
  230. new int[][] {new int[] {3, 1}}, new int[6], //
  231. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  232. new int[][] {{0}, {0}}, new int[2], new int[2], null);
  233. }
  234. @Test(expected = IllegalArgumentException.class)
  235. public void validateInputsInvalidInventories4a() {
  236. // non existing vehicle
  237. validateInputs(new int[6][6], new int[6], new int[6],
  238. new int[][] {new int[] {3, 1}}, new int[6], //
  239. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  240. new int[][] {{2, 0}, {2, 0}}, new int[2], new int[2], null);
  241. }
  242. @Test(expected = IllegalArgumentException.class)
  243. public void validateInputsInvalidInventories4b() {
  244. // non existing vehicle
  245. validateInputs(new int[6][6], new int[6], new int[6],
  246. new int[][] {new int[] {3, 1}}, new int[6], //
  247. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  248. new int[][] {{-1, 0}, {2, 0}}, new int[2], new int[2], null);
  249. }
  250. @Test(expected = IllegalArgumentException.class)
  251. public void validateInputsInvalidInventories5a() {
  252. // non existing location
  253. validateInputs(new int[6][6], new int[6], new int[6],
  254. new int[][] {new int[] {3, 1}}, new int[6], //
  255. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  256. new int[][] {{1, 0}, {1, 0}}, new int[2], new int[2], null);
  257. }
  258. @Test(expected = IllegalArgumentException.class)
  259. public void validateInputsInvalidInventories5b() {
  260. // non existing location
  261. validateInputs(new int[6][6], new int[6], new int[6],
  262. new int[][] {new int[] {3, 1}}, new int[6], //
  263. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  264. new int[][] {{1, 5}, {1, 5}}, new int[2], new int[2], null);
  265. }
  266. @Test(expected = IllegalArgumentException.class)
  267. public void validateInputsInvalidInventories6() {
  268. // reference to still available location in inventory
  269. validateInputs(new int[6][6], new int[6], new int[6],
  270. new int[][] {new int[] {3, 1}}, new int[6], //
  271. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  272. new int[][] {{1, 1}, {1, 1}}, new int[2], new int[2], null);
  273. }
  274. @Test(expected = IllegalArgumentException.class)
  275. public void validateInputsInvalidInventories7() {
  276. // duplicate inventory entry
  277. validateInputs(new int[6][6], new int[6], new int[6],
  278. new int[][] {new int[] {4, 2}}, new int[6], //
  279. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  280. new int[][] {{1, 1}, {1, 1}}, new int[2], new int[2], null);
  281. }
  282. @Test(expected = IllegalArgumentException.class)
  283. public void validateInputsInvalidRemainingServiceTimes1() {
  284. // incorrect size of array
  285. validateInputs(new int[6][6], new int[6], new int[6],
  286. new int[][] {new int[] {4, 2}}, new int[6], //
  287. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  288. new int[][] {{1, 1}, {1, 3}}, new int[] {1}, new int[2], null);
  289. }
  290. @Test(expected = IllegalArgumentException.class)
  291. public void validateInputsInvalidRemainingServiceTimes2() {
  292. // incorrect time value
  293. validateInputs(new int[6][6], new int[6], new int[6],
  294. new int[][] {new int[] {4, 2}},
  295. new int[6], //
  296. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  297. new int[][] {{1, 1}, {1, 3}}, new int[] {300, -1}, new int[2],
  298. null);
  299. }
  300. /**
  301. * Dest array length should equal v.
  302. */
  303. @Test(expected = IllegalArgumentException.class)
  304. public void validateInputsInvalidDestinations1() {
  305. validateInputs(new int[6][6], new int[6], new int[6],
  306. new int[][] {new int[] {4, 2}},
  307. new int[6], //
  308. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  309. new int[][] {{1, 1}, {1, 3}}, new int[] {300, 0}, new int[1],
  310. null);
  311. }
  312. /**
  313. * Can not have the depot as dest.
  314. */
  315. @Test(expected = IllegalArgumentException.class)
  316. public void validateInputsInvalidDestinations2a() {
  317. validateInputs(new int[6][6], new int[6], new int[6],
  318. new int[][] {new int[] {4, 2}}, new int[6], //
  319. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  320. new int[][] {{1, 1}, {1, 3}}, new int[] {300, 0}, new int[] {
  321. 5, 3},
  322. null);
  323. }
  324. /**
  325. * Can not have a negative value as dest.
  326. */
  327. @Test(expected = IllegalArgumentException.class)
  328. public void validateInputsInvalidDestinations2b() {
  329. validateInputs(new int[6][6], new int[6], new int[6],
  330. new int[][] {new int[] {4, 2}}, new int[6], //
  331. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, //
  332. new int[][] {{1, 1}, {1, 3}}, new int[] {300, 0}, new int[] {
  333. -1, 3},
  334. null);
  335. }
  336. /**
  337. * Can not have a delivery point as a destination when its corresponding
  338. * pickup point has not been visited first.
  339. */
  340. @Test(expected = IllegalArgumentException.class)
  341. public void validateInputsInvalidDestinations3() {
  342. validateInputs(new int[6][6], new int[6], new int[6],
  343. new int[][] {new int[] {4, 2}}, // servicePairs
  344. new int[6], // serviceTimes
  345. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, // vehicleTravelTimes
  346. new int[][] {{1, 3}, {1, 1}}, // inventories
  347. new int[] {300, 0}, // remainingServiceTimes
  348. new int[] {0, 2}// current destinations
  349. , null);
  350. }
  351. /**
  352. * Can not have a delivery point as a destination when it is not in *this*
  353. * vehicle's inventory.
  354. */
  355. @Test(expected = IllegalArgumentException.class)
  356. public void validateInputsInvalidDestinations4() {
  357. validateInputs(new int[6][6], new int[6], new int[6],
  358. new int[][] {new int[] {4, 2}}, // servicePairs
  359. new int[6], // serviceTimes
  360. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, // vehicleTravelTimes
  361. new int[][] {{1, 3}, {1, 1}}, // inventories
  362. new int[] {300, 0}, // remainingServiceTimes
  363. new int[] {1, 0}// current destinations
  364. , null);
  365. }
  366. /**
  367. * A vehicle can have a dest and have remaining service time.
  368. */
  369. public void validateInputsValidDestinations1() {
  370. validateInputs(new int[6][6], new int[6], new int[6],
  371. new int[][] {new int[] {4, 2}}, // servicePairs
  372. new int[6], // serviceTimes
  373. new int[][] {{0, 0, 0, 0, 0, 0}, {0, 0, 0, 0, 0, 0}}, // vehicleTravelTimes
  374. new int[][] {{1, 3}, {1, 1}}, // inventories
  375. new int[] {300, 0}, // remainingServiceTimes
  376. new int[] {4, 0}// current destinations
  377. , null);
  378. }
  379. private static final int MI = Integer.MAX_VALUE;
  380. /**
  381. * Valid destinations.
  382. */
  383. @Test
  384. public void validateInputsValidDestinations2() {
  385. validateInputs(new int[6][6], new int[6], new int[6],
  386. new int[][] {new int[] {4, 2}}, // servicePairs
  387. new int[6], // serviceTimes
  388. new int[][] {{0, MI, MI, MI, 0, MI}, {0, MI, MI, 0, MI, MI}}, // vehicleTravelTimes
  389. new int[][] {{1, 3}, {1, 1}}, // inventories
  390. new int[] {0, 0}, // remainingServiceTimes
  391. new int[] {4, 3}// current destinations
  392. , null);
  393. }
  394. @Test
  395. public void validateInputsValidMulti() {
  396. validateInputs(new int[6][6], new int[6], new int[6],
  397. new int[][] {new int[] {4, 2}}, new int[6], //
  398. new int[][] {{0, MI, MI, MI, 0, MI}, {0, 0, 0, 0, 0, 0}}, //
  399. new int[][] {{1, 1}, {1, 3}}, new int[] {300, 0}, //
  400. new int[] {4, 0}, // current destinations
  401. null);
  402. }
  403. /*
  404. * VALIDATE SOLUTION
  405. */
  406. /**
  407. * Route length must be equal to number of locations (4).
  408. */
  409. @Test(expected = IllegalArgumentException.class)
  410. public void validateInvalidRouteLength() {
  411. validateOutputs(new SolutionObject(new int[3], new int[1], 0),
  412. new int[4][4], new int[4], new int[4], new int[][] {}, new int[4],
  413. null);
  414. }
  415. /**
  416. * A location may only be visited once, duplicates are not allowed.
  417. */
  418. @Test(expected = IllegalArgumentException.class)
  419. public void validateDuplicatesInRouteStart() {
  420. validateOutputs(
  421. new SolutionObject(new int[] {1, 1, 1, 1}, new int[1], 0),
  422. new int[4][4], new int[4], new int[4], new int[][] {}, new int[4],
  423. null);
  424. }
  425. /**
  426. * The first location visited must always be the vehicle start location (0).
  427. */
  428. @Test(expected = IllegalArgumentException.class)
  429. public void validateInvalidRouteDuplicates() {
  430. validateOutputs(
  431. new SolutionObject(new int[] {1, 2, 0, 3}, new int[1], 0),
  432. new int[4][4], new int[4], new int[4], new int[][] {}, new int[4],
  433. null);
  434. }
  435. /**
  436. * The last location visited must always be the depot (n-1).
  437. */
  438. @Test(expected = IllegalArgumentException.class)
  439. public void validateInvalidRouteDepot() {
  440. validateOutputs(
  441. new SolutionObject(new int[] {0, 1, 3, 2}, new int[1], 0),
  442. new int[4][4], new int[4], new int[4], new int[][] {}, new int[4],
  443. null);
  444. }
  445. /**
  446. * All locations must be visited, can not visit non-existing locations.
  447. */
  448. @Test(expected = IllegalArgumentException.class)
  449. public void validateInvalidRouteNonExisting() {
  450. validateOutputs(
  451. new SolutionObject(new int[] {0, 1, 9, 3}, new int[1], 0),
  452. new int[4][4], new int[4], new int[4], new int[][] {}, new int[4],
  453. null);
  454. }
  455. @Test(expected = IllegalArgumentException.class)
  456. public void validateInvalidArrivalTimesLength() {
  457. validateOutputs(
  458. new SolutionObject(new int[] {0, 1, 2, 3}, new int[1], 0),
  459. new int[4][4], new int[4], new int[4], new int[][] {}, new int[4],
  460. null);
  461. }
  462. @Test(expected = IllegalArgumentException.class)
  463. public void validateInvalidArrivalTimesFirst() {
  464. validateOutputs(new SolutionObject(new int[] {0, 1, 2, 3}, new int[] {1,
  465. -1, 0, 1}, 0), new int[4][4], new int[4], new int[4], new int[][] {},
  466. new int[4], null);
  467. }
  468. @Test(expected = IllegalArgumentException.class)
  469. public void validateInvalidArrivalTimes1() {
  470. validateOutputs(new SolutionObject(new int[] {0, 1, 2, 3}, new int[] {0,
  471. -1, 0, 1}, 0), new int[4][4], new int[4], new int[4], new int[][] {},
  472. new int[4], null);
  473. }
  474. final int[][] travelTimes = new int[][] {
  475. /* */new int[] {0, 10, 999, 999},
  476. /* */new int[] {999, 0, 3, 999},
  477. /* */new int[] {999, 999, 0, 7},
  478. /* */new int[] {999, 999, 999, 0}};
  479. @Test(expected = IllegalArgumentException.class)
  480. public void validateInvalidArrivalTimes2() {
  481. validateOutputs(new SolutionObject(new int[] {0, 1, 2, 3}, new int[] {0,
  482. 0, 0, 1}, 0), travelTimes, new int[4], new int[4], new int[][] {},
  483. new int[4], null);
  484. }
  485. @Test(expected = IllegalArgumentException.class)
  486. public void validateInvalidArrivalTimes3() {
  487. validateOutputs(new SolutionObject(new int[] {0, 1, 2, 3}, new int[] {0,
  488. 10, 0, 1}, 0), travelTimes, new int[4], new int[4], new int[][] {},
  489. new int[4], null);
  490. }
  491. @Test(expected = IllegalArgumentException.class)
  492. public void validateInvalidObjective() {
  493. validateOutputs(new SolutionObject(new int[] {0, 1, 2, 3}, new int[] {0,
  494. 10, 100, 108}, 0), travelTimes, new int[4], new int[4],
  495. new int[][] {}, new int[4], null);
  496. }
  497. /*
  498. * VALIDATE MULTI VEHICLE
  499. */
  500. @Test(expected = IllegalArgumentException.class)
  501. public void validateMissingLocation2Vehicles() {
  502. final SolutionObject sol1 = new SolutionObject(new int[] {0, 2, 3},
  503. new int[] {0, 10, 100, 108}, 238);
  504. final SolutionObject sol2 = new SolutionObject(new int[] {0, 2, 3},
  505. new int[] {0, 10, 100, 108}, 238);
  506. validateOutputs(new SolutionObject[] {sol1, sol2}, travelTimes,
  507. new int[4], new int[4], new int[][] {}, new int[4], new int[2][4],
  508. new int[2][2], new int[2], new int[2]);
  509. }
  510. @Test
  511. public void validateMultiVehicleCorrect() {
  512. final int[][] travelTime = new int[][] {
  513. /* */new int[] {999, 999, 999, 999},
  514. /* */new int[] {999, 0, 999, 3},
  515. /* */new int[] {999, 999, 0, 7},
  516. /* */new int[] {999, 999, 999, 0}};
  517. final int[] releaseDates = {0, 7, 8, 0};
  518. final int[] dueDates = {0, 9, 100, 40};
  519. final int[][] vehicleTravelTime = new int[][] {
  520. /* */new int[] {999, 9, 999, 999},
  521. /* */new int[] {999, 999, 2, 999}};
  522. // travel time for this route: 9 + 3
  523. // tardiness: 1 + 60
  524. final SolutionObject sol1 = new SolutionObject(new int[] {0, 1, 3},
  525. new int[] {0, 10, 100}, 73);
  526. // travel time for this route: 2 + 7
  527. // tardiness: 0 + 50
  528. final SolutionObject sol2 = new SolutionObject(new int[] {0, 2, 3},
  529. new int[] {0, 15, 90}, 59);
  530. validateOutputs(new SolutionObject[] {sol1, sol2}, travelTime,
  531. releaseDates, dueDates, new int[][] {}, new int[4], vehicleTravelTime,
  532. new int[2][2], new int[2], new int[2]);
  533. }
  534. /**
  535. * In this test the vehicles attempt to deliver parcels which are not in their
  536. * own inventory.
  537. */
  538. @Test(expected = IllegalArgumentException.class)
  539. public void validateRouteWithoutInventoryLocations() {
  540. final SolutionObject sol1 = new SolutionObject(new int[] {0, 1, 3},
  541. new int[] {0, 10, 100}, 73);
  542. final SolutionObject sol2 = new SolutionObject(new int[] {0, 2, 3},
  543. new int[] {0, 15, 90}, 59);
  544. // vehicle 0 has location 2
  545. // vehicle 1 has location 1
  546. final int[][] inventories = new int[][] {{0, 2}, {1, 1}};
  547. validateOutputs(new SolutionObject[] {sol1, sol2}, new int[4][4],
  548. new int[4], new int[4], new int[][] {}, new int[4], new int[2][4],
  549. inventories, new int[2], new int[2]);
  550. }
  551. /**
  552. * The first arrival time should reflect the remainingServiceTime for each
  553. * vehicle.
  554. */
  555. @Test(expected = IllegalArgumentException.class)
  556. public void validateRemainingServiceTime() {
  557. final SolutionObject sol1 = new SolutionObject(new int[] {0, 1, 3},
  558. new int[] {0, 10, 100}, 73);
  559. final SolutionObject sol2 = new SolutionObject(new int[] {0, 2, 3},
  560. new int[] {0, 15, 90}, 59);
  561. final int[] remainingServiceTimes = new int[] {3, 700};
  562. validateOutputs(new SolutionObject[] {sol1, sol2}, new int[4][4],
  563. new int[4], new int[4], new int[][] {}, new int[4], new int[2][4],
  564. new int[0][0], remainingServiceTimes, new int[2]);
  565. }
  566. /**
  567. * The first point to visit in a route must be the destination (if there is a
  568. * destination).
  569. */
  570. @Test(expected = IllegalArgumentException.class)
  571. public void validateCurrentDestination() {
  572. final SolutionObject sol1 = new SolutionObject(new int[] {0, 2, 3},
  573. new int[] {0, 10, 100}, 73);
  574. final SolutionObject sol2 = new SolutionObject(new int[] {0, 1, 3},
  575. new int[] {0, 15, 90}, 59);
  576. final int[] remainingServiceTimes = new int[] {0, 0};
  577. validateOutputs(new SolutionObject[] {sol1, sol2}, new int[4][4],
  578. new int[4], new int[4], new int[][] {}, new int[4], new int[2][4],
  579. new int[0][0], remainingServiceTimes, new int[] {1, 2});
  580. }
  581. @Test
  582. public void validateCorrect() {
  583. validateOutputs(
  584. new SolutionObject(new int[] {0, 1, 2, 3}, new int[] {0, 10, 100,
  585. 108}, 238),
  586. travelTimes, new int[4], new int[4], new int[][] {},
  587. new int[4], null).toString();
  588. }
  589. @Test
  590. public void validateCorrect2() {
  591. final int[][] tt = new int[][] {
  592. /* */new int[] {999, 999, 999, 999},
  593. /* */new int[] {999, 0, 3, 999},
  594. /* */new int[] {999, 999, 0, 7},
  595. /* */new int[] {999, 999, 999, 0}};
  596. // travel time = 0 + 3 + 7 = 10
  597. // tardiness = 10 + 100 + 108 = 218
  598. // objval = 228
  599. final SolutionObject sol1 = new SolutionObject(//
  600. new int[] {0, 1, 2, 3}, // route
  601. new int[] {0, 10, 100, 108}, // arrival times
  602. 228);// objective value
  603. // travel time = 0
  604. // tardiness = 999
  605. // objval = 999
  606. final SolutionObject sol2 = new SolutionObject(//
  607. new int[] {0, 3}, // route
  608. new int[] {0, 999}, // arrival times
  609. 999);// objective value
  610. validateOutputs(new SolutionObject[] {sol1, sol2}, tt, new int[4],
  611. new int[4], new int[][] {}, new int[4], new int[2][4], new int[2][2],
  612. new int[2], new int[2]);
  613. }
  614. /**
  615. * Valid test with currentDestinations set.
  616. */
  617. @Test
  618. public void validateCorrect3() {
  619. final int[][] tt = new int[][] {
  620. /* */new int[] {999, 999, 999, 999, 999},
  621. /* */new int[] {999, 0, 3, 999, 60},
  622. /* */new int[] {999, 999, 0, 7, 999},
  623. /* */new int[] {999, 999, 0, 7, 50},
  624. /* */new int[] {999, 999, 999, 0, 999}};
  625. final SolutionObject sol1 = new SolutionObject(//
  626. new int[] {0, 1, 4}, // route
  627. new int[] {0, 10, 70}, // arrival times
  628. 140);// objective value
  629. final SolutionObject sol2 = new SolutionObject(//
  630. new int[] {0, 2, 3, 4}, // route
  631. new int[] {0, 10, 20, 108}, // arrival times
  632. 195);// objective value
  633. final MultiVehicleArraysSolver s = ArraysSolverValidator
  634. .wrap(new FakeMultiSolver(new SolutionObject[] {sol1, sol2}));
  635. s.solve(tt, // travel times
  636. new int[5], // release dates
  637. new int[5], // due dates
  638. new int[][] {{2, 3}}, // service pairs
  639. new int[5], // service times
  640. // vehicle travel times
  641. new int[][] {{0, 0, MI, MI, MI}, {0, MI, 0, MI, MI}}, //
  642. new int[][] {{0, 1}}, // inventories
  643. new int[2], // remaining service times
  644. new int[] {1, 2}, // current destinations
  645. null);
  646. }
  647. @Test(expected = IllegalArgumentException.class)
  648. public void validateBig() {
  649. final int[][] travelTime = new int[][] {
  650. {0, 132, 233, 491, 644, 513, 284, 447, 435, 255, 140, 246},
  651. {132, 0, 168, 380, 548, 508, 228, 399, 394, 301, 161, 163},
  652. {233, 168, 0, 276, 414, 351, 61, 232, 227, 216, 127, 35},
  653. {491, 380, 276, 0, 185, 439, 256, 263, 282, 463, 402, 250},
  654. {644, 548, 414, 185, 0, 427, 373, 283, 307, 555, 534, 399},
  655. {513, 508, 351, 439, 427, 0, 297, 177, 160, 280, 376, 373},
  656. {284, 228, 61, 256, 373, 297, 0, 172, 167, 210, 162, 77},
  657. {447, 399, 232, 263, 283, 177, 172, 0, 25, 286, 312, 242},
  658. {435, 394, 227, 282, 307, 160, 167, 25, 0, 265, 298, 240},
  659. {255, 301, 216, 463, 555, 280, 210, 286, 265, 0, 141, 250},
  660. {140, 161, 127, 402, 534, 376, 162, 312, 298, 141, 0, 155},
  661. {246, 163, 35, 250, 399, 373, 77, 242, 240, 250, 155, 0}};
  662. final int[] releaseDates = {0, 1245143, 1755843, 63643, 1330543, 711843,
  663. 1811843, 212643, 1453443, 0, 188143, 0};
  664. final int[] dueDates = {0, 2139143, 2606143, 1758143, 2242143, 1968143,
  665. 2564143, 1395843, 2401143, 1843643, 2030043, 2940143};
  666. final int[][] servicePairs = {{1, 2}, {3, 4}, {5, 6}, {7, 8}};
  667. final int[] serviceTimes = createServiceTimes(300, releaseDates.length);
  668. // Route: [0, 1, 10, 9, 5, 8, 7, 4, 3, 6, 2, 11]
  669. // Arrival times: [0, 132, 4384, 3467, 2982, 1614, 4023, 2399, 2074,
  670. // 1034, 593, 4719]
  671. // Objective: 1719
  672. final SolutionObject sol = new SolutionObject(//
  673. new int[] {0, 1, 10, 9, 5, 8, 7, 4, 3, 6, 2, 11}, //
  674. new int[] {0, 132, 4384, 3467, 2982, 1614, 4023, 2399, 2074, 1034,
  675. 593, 4719}, //
  676. 1719);
  677. validateOutputs(sol, travelTime, releaseDates, dueDates, servicePairs,
  678. serviceTimes, null);
  679. }
  680. @Test(expected = IllegalArgumentException.class)
  681. public void validateInvalidArrivalOrder() {
  682. validateOutputs(
  683. new SolutionObject(//
  684. new int[] {0, 2, 1, 3}, // route
  685. new int[] {0, 1998, 999, 2997}, // arrival times
  686. 238), // obj. value
  687. travelTimes, new int[4], new int[4], new int[][] {{1, 2}},
  688. new int[4], null);
  689. }
  690. @Test
  691. public void testWrapSingle() {
  692. final SingleVehicleArraysSolver s = wrap(new FakeSingleSolver(
  693. new SolutionObject(new int[] {0, 1, 2, 3}, new int[] {0, 10, 100,
  694. 108}, 238)));
  695. s.solve(travelTimes, new int[4], new int[4], new int[][] {}, new int[4],
  696. null);
  697. }
  698. @Test
  699. public void testWrapMulti() {
  700. TestUtil.testPrivateConstructor(ArraysSolverValidator.class);
  701. final MultiVehicleArraysSolver s = wrap(new FakeMultiSolver(
  702. new SolutionObject[] {new SolutionObject(new int[] {0, 1, 2, 3},
  703. new int[] {0, 10, 100, 108}, 228)}));
  704. s.solve(travelTimes, new int[4], new int[4], new int[][] {}, new int[4],
  705. new int[2][4], new int[][] {{0, 1}, {0, 2}}, new int[2],
  706. new int[2], null);
  707. }
  708. class FakeSingleSolver implements SingleVehicleArraysSolver {
  709. SolutionObject answer;
  710. FakeSingleSolver(SolutionObject answer) {
  711. this.answer = answer;
  712. }
  713. @Override
  714. public SolutionObject solve(int[][] travelTime, int[] releaseDates,
  715. int[] dueDates, int[][] servicePairs, int serviceTimes[],
  716. @Nullable SolutionObject currentSolution) {
  717. return answer;
  718. }
  719. }
  720. class FakeMultiSolver implements MultiVehicleArraysSolver {
  721. SolutionObject[] answer;
  722. FakeMultiSolver(SolutionObject[] answer) {
  723. this.answer = answer;
  724. }
  725. @Override
  726. public SolutionObject[] solve(int[][] travelTime, int[] releaseDates,
  727. int[] dueDates, int[][] servicePairs, int[] serviceTimes,
  728. int[][] vehicleTravelTimes, int[][] inventories,
  729. int[] remainingServiceTimes, int[] currentDestinations,
  730. @Nullable SolutionObject[] currentSolutions) {
  731. return answer;
  732. }
  733. }
  734. public static int[] createServiceTimes(int val, int length) {
  735. checkArgument(length > 2);
  736. final int[] serviceTimes = new int[length];
  737. Arrays.fill(serviceTimes, val);
  738. serviceTimes[0] = 0;
  739. serviceTimes[length - 1] = 0;
  740. return serviceTimes;
  741. }
  742. }