test_densetools.py 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714
  1. """Tests for dense recursive polynomials' tools. """
  2. from sympy.polys.densebasic import (
  3. dup_normal, dmp_normal,
  4. dup_from_raw_dict,
  5. dmp_convert, dmp_swap,
  6. )
  7. from sympy.polys.densearith import dmp_mul_ground
  8. from sympy.polys.densetools import (
  9. dup_clear_denoms, dmp_clear_denoms,
  10. dup_integrate, dmp_integrate, dmp_integrate_in,
  11. dup_diff, dmp_diff, dmp_diff_in,
  12. dup_eval, dmp_eval, dmp_eval_in,
  13. dmp_eval_tail, dmp_diff_eval_in,
  14. dup_trunc, dmp_trunc, dmp_ground_trunc,
  15. dup_monic, dmp_ground_monic,
  16. dup_content, dmp_ground_content,
  17. dup_primitive, dmp_ground_primitive,
  18. dup_extract, dmp_ground_extract,
  19. dup_real_imag,
  20. dup_mirror, dup_scale, dup_shift, dmp_shift,
  21. dup_transform,
  22. dup_compose, dmp_compose,
  23. dup_decompose,
  24. dmp_lift,
  25. dup_sign_variations,
  26. dup_revert, dmp_revert,
  27. )
  28. from sympy.polys.polyclasses import ANP
  29. from sympy.polys.polyerrors import (
  30. MultivariatePolynomialError,
  31. ExactQuotientFailed,
  32. NotReversible,
  33. DomainError,
  34. )
  35. from sympy.polys.specialpolys import f_polys
  36. from sympy.polys.domains import FF, ZZ, QQ, ZZ_I, QQ_I, EX, RR
  37. from sympy.polys.rings import ring
  38. from sympy.core.numbers import I
  39. from sympy.core.singleton import S
  40. from sympy.functions.elementary.trigonometric import sin
  41. from sympy.abc import x
  42. from sympy.testing.pytest import raises
  43. f_0, f_1, f_2, f_3, f_4, f_5, f_6 = [ f.to_dense() for f in f_polys() ]
  44. def test_dup_integrate():
  45. assert dup_integrate([], 1, QQ) == []
  46. assert dup_integrate([], 2, QQ) == []
  47. assert dup_integrate([QQ(1)], 1, QQ) == [QQ(1), QQ(0)]
  48. assert dup_integrate([QQ(1)], 2, QQ) == [QQ(1, 2), QQ(0), QQ(0)]
  49. assert dup_integrate([QQ(1), QQ(2), QQ(3)], 0, QQ) == \
  50. [QQ(1), QQ(2), QQ(3)]
  51. assert dup_integrate([QQ(1), QQ(2), QQ(3)], 1, QQ) == \
  52. [QQ(1, 3), QQ(1), QQ(3), QQ(0)]
  53. assert dup_integrate([QQ(1), QQ(2), QQ(3)], 2, QQ) == \
  54. [QQ(1, 12), QQ(1, 3), QQ(3, 2), QQ(0), QQ(0)]
  55. assert dup_integrate([QQ(1), QQ(2), QQ(3)], 3, QQ) == \
  56. [QQ(1, 60), QQ(1, 12), QQ(1, 2), QQ(0), QQ(0), QQ(0)]
  57. assert dup_integrate(dup_from_raw_dict({29: QQ(17)}, QQ), 3, QQ) == \
  58. dup_from_raw_dict({32: QQ(17, 29760)}, QQ)
  59. assert dup_integrate(dup_from_raw_dict({29: QQ(17), 5: QQ(1, 2)}, QQ), 3, QQ) == \
  60. dup_from_raw_dict({32: QQ(17, 29760), 8: QQ(1, 672)}, QQ)
  61. def test_dmp_integrate():
  62. assert dmp_integrate([QQ(1)], 2, 0, QQ) == [QQ(1, 2), QQ(0), QQ(0)]
  63. assert dmp_integrate([[[]]], 1, 2, QQ) == [[[]]]
  64. assert dmp_integrate([[[]]], 2, 2, QQ) == [[[]]]
  65. assert dmp_integrate([[[QQ(1)]]], 1, 2, QQ) == [[[QQ(1)]], [[]]]
  66. assert dmp_integrate([[[QQ(1)]]], 2, 2, QQ) == [[[QQ(1, 2)]], [[]], [[]]]
  67. assert dmp_integrate([[QQ(1)], [QQ(2)], [QQ(3)]], 0, 1, QQ) == \
  68. [[QQ(1)], [QQ(2)], [QQ(3)]]
  69. assert dmp_integrate([[QQ(1)], [QQ(2)], [QQ(3)]], 1, 1, QQ) == \
  70. [[QQ(1, 3)], [QQ(1)], [QQ(3)], []]
  71. assert dmp_integrate([[QQ(1)], [QQ(2)], [QQ(3)]], 2, 1, QQ) == \
  72. [[QQ(1, 12)], [QQ(1, 3)], [QQ(3, 2)], [], []]
  73. assert dmp_integrate([[QQ(1)], [QQ(2)], [QQ(3)]], 3, 1, QQ) == \
  74. [[QQ(1, 60)], [QQ(1, 12)], [QQ(1, 2)], [], [], []]
  75. def test_dmp_integrate_in():
  76. f = dmp_convert(f_6, 3, ZZ, QQ)
  77. assert dmp_integrate_in(f, 2, 1, 3, QQ) == \
  78. dmp_swap(
  79. dmp_integrate(dmp_swap(f, 0, 1, 3, QQ), 2, 3, QQ), 0, 1, 3, QQ)
  80. assert dmp_integrate_in(f, 3, 1, 3, QQ) == \
  81. dmp_swap(
  82. dmp_integrate(dmp_swap(f, 0, 1, 3, QQ), 3, 3, QQ), 0, 1, 3, QQ)
  83. assert dmp_integrate_in(f, 2, 2, 3, QQ) == \
  84. dmp_swap(
  85. dmp_integrate(dmp_swap(f, 0, 2, 3, QQ), 2, 3, QQ), 0, 2, 3, QQ)
  86. assert dmp_integrate_in(f, 3, 2, 3, QQ) == \
  87. dmp_swap(
  88. dmp_integrate(dmp_swap(f, 0, 2, 3, QQ), 3, 3, QQ), 0, 2, 3, QQ)
  89. raises(IndexError, lambda: dmp_integrate_in(f, 1, -1, 3, QQ))
  90. raises(IndexError, lambda: dmp_integrate_in(f, 1, 4, 3, QQ))
  91. def test_dup_diff():
  92. assert dup_diff([], 1, ZZ) == []
  93. assert dup_diff([7], 1, ZZ) == []
  94. assert dup_diff([2, 7], 1, ZZ) == [2]
  95. assert dup_diff([1, 2, 1], 1, ZZ) == [2, 2]
  96. assert dup_diff([1, 2, 3, 4], 1, ZZ) == [3, 4, 3]
  97. assert dup_diff([1, -1, 0, 0, 2], 1, ZZ) == [4, -3, 0, 0]
  98. f = dup_normal([17, 34, 56, -345, 23, 76, 0, 0, 12, 3, 7], ZZ)
  99. assert dup_diff(f, 0, ZZ) == f
  100. assert dup_diff(f, 1, ZZ) == [170, 306, 448, -2415, 138, 380, 0, 0, 24, 3]
  101. assert dup_diff(f, 2, ZZ) == dup_diff(dup_diff(f, 1, ZZ), 1, ZZ)
  102. assert dup_diff(
  103. f, 3, ZZ) == dup_diff(dup_diff(dup_diff(f, 1, ZZ), 1, ZZ), 1, ZZ)
  104. K = FF(3)
  105. f = dup_normal([17, 34, 56, -345, 23, 76, 0, 0, 12, 3, 7], K)
  106. assert dup_diff(f, 1, K) == dup_normal([2, 0, 1, 0, 0, 2, 0, 0, 0, 0], K)
  107. assert dup_diff(f, 2, K) == dup_normal([1, 0, 0, 2, 0, 0, 0], K)
  108. assert dup_diff(f, 3, K) == dup_normal([], K)
  109. assert dup_diff(f, 0, K) == f
  110. assert dup_diff(f, 2, K) == dup_diff(dup_diff(f, 1, K), 1, K)
  111. assert dup_diff(
  112. f, 3, K) == dup_diff(dup_diff(dup_diff(f, 1, K), 1, K), 1, K)
  113. def test_dmp_diff():
  114. assert dmp_diff([], 1, 0, ZZ) == []
  115. assert dmp_diff([[]], 1, 1, ZZ) == [[]]
  116. assert dmp_diff([[[]]], 1, 2, ZZ) == [[[]]]
  117. assert dmp_diff([[[1], [2]]], 1, 2, ZZ) == [[[]]]
  118. assert dmp_diff([[[1]], [[]]], 1, 2, ZZ) == [[[1]]]
  119. assert dmp_diff([[[3]], [[1]], [[]]], 1, 2, ZZ) == [[[6]], [[1]]]
  120. assert dmp_diff([1, -1, 0, 0, 2], 1, 0, ZZ) == \
  121. dup_diff([1, -1, 0, 0, 2], 1, ZZ)
  122. assert dmp_diff(f_6, 0, 3, ZZ) == f_6
  123. assert dmp_diff(f_6, 1, 3, ZZ) == [[[[8460]], [[]]],
  124. [[[135, 0, 0], [], [], [-135, 0, 0]]],
  125. [[[]]],
  126. [[[-423]], [[-47]], [[]], [[141], [], [94, 0], []], [[]]]]
  127. assert dmp_diff(
  128. f_6, 2, 3, ZZ) == dmp_diff(dmp_diff(f_6, 1, 3, ZZ), 1, 3, ZZ)
  129. assert dmp_diff(f_6, 3, 3, ZZ) == dmp_diff(
  130. dmp_diff(dmp_diff(f_6, 1, 3, ZZ), 1, 3, ZZ), 1, 3, ZZ)
  131. K = FF(23)
  132. F_6 = dmp_normal(f_6, 3, K)
  133. assert dmp_diff(F_6, 0, 3, K) == F_6
  134. assert dmp_diff(F_6, 1, 3, K) == dmp_diff(F_6, 1, 3, K)
  135. assert dmp_diff(F_6, 2, 3, K) == dmp_diff(dmp_diff(F_6, 1, 3, K), 1, 3, K)
  136. assert dmp_diff(F_6, 3, 3, K) == dmp_diff(
  137. dmp_diff(dmp_diff(F_6, 1, 3, K), 1, 3, K), 1, 3, K)
  138. def test_dmp_diff_in():
  139. assert dmp_diff_in(f_6, 2, 1, 3, ZZ) == \
  140. dmp_swap(dmp_diff(dmp_swap(f_6, 0, 1, 3, ZZ), 2, 3, ZZ), 0, 1, 3, ZZ)
  141. assert dmp_diff_in(f_6, 3, 1, 3, ZZ) == \
  142. dmp_swap(dmp_diff(dmp_swap(f_6, 0, 1, 3, ZZ), 3, 3, ZZ), 0, 1, 3, ZZ)
  143. assert dmp_diff_in(f_6, 2, 2, 3, ZZ) == \
  144. dmp_swap(dmp_diff(dmp_swap(f_6, 0, 2, 3, ZZ), 2, 3, ZZ), 0, 2, 3, ZZ)
  145. assert dmp_diff_in(f_6, 3, 2, 3, ZZ) == \
  146. dmp_swap(dmp_diff(dmp_swap(f_6, 0, 2, 3, ZZ), 3, 3, ZZ), 0, 2, 3, ZZ)
  147. raises(IndexError, lambda: dmp_diff_in(f_6, 1, -1, 3, ZZ))
  148. raises(IndexError, lambda: dmp_diff_in(f_6, 1, 4, 3, ZZ))
  149. def test_dup_eval():
  150. assert dup_eval([], 7, ZZ) == 0
  151. assert dup_eval([1, 2], 0, ZZ) == 2
  152. assert dup_eval([1, 2, 3], 7, ZZ) == 66
  153. def test_dmp_eval():
  154. assert dmp_eval([], 3, 0, ZZ) == 0
  155. assert dmp_eval([[]], 3, 1, ZZ) == []
  156. assert dmp_eval([[[]]], 3, 2, ZZ) == [[]]
  157. assert dmp_eval([[1, 2]], 0, 1, ZZ) == [1, 2]
  158. assert dmp_eval([[[1]]], 3, 2, ZZ) == [[1]]
  159. assert dmp_eval([[[1, 2]]], 3, 2, ZZ) == [[1, 2]]
  160. assert dmp_eval([[3, 2], [1, 2]], 3, 1, ZZ) == [10, 8]
  161. assert dmp_eval([[[3, 2]], [[1, 2]]], 3, 2, ZZ) == [[10, 8]]
  162. def test_dmp_eval_in():
  163. assert dmp_eval_in(
  164. f_6, -2, 1, 3, ZZ) == dmp_eval(dmp_swap(f_6, 0, 1, 3, ZZ), -2, 3, ZZ)
  165. assert dmp_eval_in(
  166. f_6, 7, 1, 3, ZZ) == dmp_eval(dmp_swap(f_6, 0, 1, 3, ZZ), 7, 3, ZZ)
  167. assert dmp_eval_in(f_6, -2, 2, 3, ZZ) == dmp_swap(
  168. dmp_eval(dmp_swap(f_6, 0, 2, 3, ZZ), -2, 3, ZZ), 0, 1, 2, ZZ)
  169. assert dmp_eval_in(f_6, 7, 2, 3, ZZ) == dmp_swap(
  170. dmp_eval(dmp_swap(f_6, 0, 2, 3, ZZ), 7, 3, ZZ), 0, 1, 2, ZZ)
  171. f = [[[int(45)]], [[]], [[]], [[int(-9)], [-1], [], [int(3), int(0), int(10), int(0)]]]
  172. assert dmp_eval_in(f, -2, 2, 2, ZZ) == \
  173. [[45], [], [], [-9, -1, 0, -44]]
  174. raises(IndexError, lambda: dmp_eval_in(f_6, ZZ(1), -1, 3, ZZ))
  175. raises(IndexError, lambda: dmp_eval_in(f_6, ZZ(1), 4, 3, ZZ))
  176. def test_dmp_eval_tail():
  177. assert dmp_eval_tail([[]], [1], 1, ZZ) == []
  178. assert dmp_eval_tail([[[]]], [1], 2, ZZ) == [[]]
  179. assert dmp_eval_tail([[[]]], [1, 2], 2, ZZ) == []
  180. assert dmp_eval_tail(f_0, [], 2, ZZ) == f_0
  181. assert dmp_eval_tail(f_0, [1, -17, 8], 2, ZZ) == 84496
  182. assert dmp_eval_tail(f_0, [-17, 8], 2, ZZ) == [-1409, 3, 85902]
  183. assert dmp_eval_tail(f_0, [8], 2, ZZ) == [[83, 2], [3], [302, 81, 1]]
  184. assert dmp_eval_tail(f_1, [-17, 8], 2, ZZ) == [-136, 15699, 9166, -27144]
  185. assert dmp_eval_tail(
  186. f_2, [-12, 3], 2, ZZ) == [-1377, 0, -702, -1224, 0, -624]
  187. assert dmp_eval_tail(
  188. f_3, [-12, 3], 2, ZZ) == [144, 82, -5181, -28872, -14868, -540]
  189. assert dmp_eval_tail(
  190. f_4, [25, -1], 2, ZZ) == [152587890625, 9765625, -59605407714843750,
  191. -3839159765625, -1562475, 9536712644531250, 610349546750, -4, 24414375000, 1562520]
  192. assert dmp_eval_tail(f_5, [25, -1], 2, ZZ) == [-1, -78, -2028, -17576]
  193. assert dmp_eval_tail(f_6, [0, 2, 4], 3, ZZ) == [5040, 0, 0, 4480]
  194. def test_dmp_diff_eval_in():
  195. assert dmp_diff_eval_in(f_6, 2, 7, 1, 3, ZZ) == \
  196. dmp_eval(dmp_diff(dmp_swap(f_6, 0, 1, 3, ZZ), 2, 3, ZZ), 7, 3, ZZ)
  197. assert dmp_diff_eval_in(f_6, 2, 7, 0, 3, ZZ) == \
  198. dmp_eval(dmp_diff(f_6, 2, 3, ZZ), 7, 3, ZZ)
  199. raises(IndexError, lambda: dmp_diff_eval_in(f_6, 1, ZZ(1), 4, 3, ZZ))
  200. def test_dup_revert():
  201. f = [-QQ(1, 720), QQ(0), QQ(1, 24), QQ(0), -QQ(1, 2), QQ(0), QQ(1)]
  202. g = [QQ(61, 720), QQ(0), QQ(5, 24), QQ(0), QQ(1, 2), QQ(0), QQ(1)]
  203. assert dup_revert(f, 8, QQ) == g
  204. raises(NotReversible, lambda: dup_revert([QQ(1), QQ(0)], 3, QQ))
  205. def test_dmp_revert():
  206. f = [-QQ(1, 720), QQ(0), QQ(1, 24), QQ(0), -QQ(1, 2), QQ(0), QQ(1)]
  207. g = [QQ(61, 720), QQ(0), QQ(5, 24), QQ(0), QQ(1, 2), QQ(0), QQ(1)]
  208. assert dmp_revert(f, 8, 0, QQ) == g
  209. raises(MultivariatePolynomialError, lambda: dmp_revert([[1]], 2, 1, QQ))
  210. def test_dup_trunc():
  211. assert dup_trunc([1, 2, 3, 4, 5, 6], ZZ(3), ZZ) == [1, -1, 0, 1, -1, 0]
  212. assert dup_trunc([6, 5, 4, 3, 2, 1], ZZ(3), ZZ) == [-1, 1, 0, -1, 1]
  213. R = ZZ_I
  214. assert dup_trunc([R(3), R(4), R(5)], R(3), R) == [R(1), R(-1)]
  215. K = FF(5)
  216. assert dup_trunc([K(3), K(4), K(5)], K(3), K) == [K(1), K(0)]
  217. def test_dmp_trunc():
  218. assert dmp_trunc([[]], [1, 2], 2, ZZ) == [[]]
  219. assert dmp_trunc([[1, 2], [1, 4, 1], [1]], [1, 2], 1, ZZ) == [[-3], [1]]
  220. def test_dmp_ground_trunc():
  221. assert dmp_ground_trunc(f_0, ZZ(3), 2, ZZ) == \
  222. dmp_normal(
  223. [[[1, -1, 0], [-1]], [[]], [[1, -1, 0], [1, -1, 1], [1]]], 2, ZZ)
  224. def test_dup_monic():
  225. assert dup_monic([3, 6, 9], ZZ) == [1, 2, 3]
  226. raises(ExactQuotientFailed, lambda: dup_monic([3, 4, 5], ZZ))
  227. assert dup_monic([], QQ) == []
  228. assert dup_monic([QQ(1)], QQ) == [QQ(1)]
  229. assert dup_monic([QQ(7), QQ(1), QQ(21)], QQ) == [QQ(1), QQ(1, 7), QQ(3)]
  230. def test_dmp_ground_monic():
  231. assert dmp_ground_monic([3, 6, 9], 0, ZZ) == [1, 2, 3]
  232. assert dmp_ground_monic([[3], [6], [9]], 1, ZZ) == [[1], [2], [3]]
  233. raises(
  234. ExactQuotientFailed, lambda: dmp_ground_monic([[3], [4], [5]], 1, ZZ))
  235. assert dmp_ground_monic([[]], 1, QQ) == [[]]
  236. assert dmp_ground_monic([[QQ(1)]], 1, QQ) == [[QQ(1)]]
  237. assert dmp_ground_monic(
  238. [[QQ(7)], [QQ(1)], [QQ(21)]], 1, QQ) == [[QQ(1)], [QQ(1, 7)], [QQ(3)]]
  239. def test_dup_content():
  240. assert dup_content([], ZZ) == ZZ(0)
  241. assert dup_content([1], ZZ) == ZZ(1)
  242. assert dup_content([-1], ZZ) == ZZ(1)
  243. assert dup_content([1, 1], ZZ) == ZZ(1)
  244. assert dup_content([2, 2], ZZ) == ZZ(2)
  245. assert dup_content([1, 2, 1], ZZ) == ZZ(1)
  246. assert dup_content([2, 4, 2], ZZ) == ZZ(2)
  247. assert dup_content([QQ(2, 3), QQ(4, 9)], QQ) == QQ(2, 9)
  248. assert dup_content([QQ(2, 3), QQ(4, 5)], QQ) == QQ(2, 15)
  249. def test_dmp_ground_content():
  250. assert dmp_ground_content([[]], 1, ZZ) == ZZ(0)
  251. assert dmp_ground_content([[]], 1, QQ) == QQ(0)
  252. assert dmp_ground_content([[1]], 1, ZZ) == ZZ(1)
  253. assert dmp_ground_content([[-1]], 1, ZZ) == ZZ(1)
  254. assert dmp_ground_content([[1], [1]], 1, ZZ) == ZZ(1)
  255. assert dmp_ground_content([[2], [2]], 1, ZZ) == ZZ(2)
  256. assert dmp_ground_content([[1], [2], [1]], 1, ZZ) == ZZ(1)
  257. assert dmp_ground_content([[2], [4], [2]], 1, ZZ) == ZZ(2)
  258. assert dmp_ground_content([[QQ(2, 3)], [QQ(4, 9)]], 1, QQ) == QQ(2, 9)
  259. assert dmp_ground_content([[QQ(2, 3)], [QQ(4, 5)]], 1, QQ) == QQ(2, 15)
  260. assert dmp_ground_content(f_0, 2, ZZ) == ZZ(1)
  261. assert dmp_ground_content(
  262. dmp_mul_ground(f_0, ZZ(2), 2, ZZ), 2, ZZ) == ZZ(2)
  263. assert dmp_ground_content(f_1, 2, ZZ) == ZZ(1)
  264. assert dmp_ground_content(
  265. dmp_mul_ground(f_1, ZZ(3), 2, ZZ), 2, ZZ) == ZZ(3)
  266. assert dmp_ground_content(f_2, 2, ZZ) == ZZ(1)
  267. assert dmp_ground_content(
  268. dmp_mul_ground(f_2, ZZ(4), 2, ZZ), 2, ZZ) == ZZ(4)
  269. assert dmp_ground_content(f_3, 2, ZZ) == ZZ(1)
  270. assert dmp_ground_content(
  271. dmp_mul_ground(f_3, ZZ(5), 2, ZZ), 2, ZZ) == ZZ(5)
  272. assert dmp_ground_content(f_4, 2, ZZ) == ZZ(1)
  273. assert dmp_ground_content(
  274. dmp_mul_ground(f_4, ZZ(6), 2, ZZ), 2, ZZ) == ZZ(6)
  275. assert dmp_ground_content(f_5, 2, ZZ) == ZZ(1)
  276. assert dmp_ground_content(
  277. dmp_mul_ground(f_5, ZZ(7), 2, ZZ), 2, ZZ) == ZZ(7)
  278. assert dmp_ground_content(f_6, 3, ZZ) == ZZ(1)
  279. assert dmp_ground_content(
  280. dmp_mul_ground(f_6, ZZ(8), 3, ZZ), 3, ZZ) == ZZ(8)
  281. def test_dup_primitive():
  282. assert dup_primitive([], ZZ) == (ZZ(0), [])
  283. assert dup_primitive([ZZ(1)], ZZ) == (ZZ(1), [ZZ(1)])
  284. assert dup_primitive([ZZ(1), ZZ(1)], ZZ) == (ZZ(1), [ZZ(1), ZZ(1)])
  285. assert dup_primitive([ZZ(2), ZZ(2)], ZZ) == (ZZ(2), [ZZ(1), ZZ(1)])
  286. assert dup_primitive(
  287. [ZZ(1), ZZ(2), ZZ(1)], ZZ) == (ZZ(1), [ZZ(1), ZZ(2), ZZ(1)])
  288. assert dup_primitive(
  289. [ZZ(2), ZZ(4), ZZ(2)], ZZ) == (ZZ(2), [ZZ(1), ZZ(2), ZZ(1)])
  290. assert dup_primitive([], QQ) == (QQ(0), [])
  291. assert dup_primitive([QQ(1)], QQ) == (QQ(1), [QQ(1)])
  292. assert dup_primitive([QQ(1), QQ(1)], QQ) == (QQ(1), [QQ(1), QQ(1)])
  293. assert dup_primitive([QQ(2), QQ(2)], QQ) == (QQ(2), [QQ(1), QQ(1)])
  294. assert dup_primitive(
  295. [QQ(1), QQ(2), QQ(1)], QQ) == (QQ(1), [QQ(1), QQ(2), QQ(1)])
  296. assert dup_primitive(
  297. [QQ(2), QQ(4), QQ(2)], QQ) == (QQ(2), [QQ(1), QQ(2), QQ(1)])
  298. assert dup_primitive(
  299. [QQ(2, 3), QQ(4, 9)], QQ) == (QQ(2, 9), [QQ(3), QQ(2)])
  300. assert dup_primitive(
  301. [QQ(2, 3), QQ(4, 5)], QQ) == (QQ(2, 15), [QQ(5), QQ(6)])
  302. def test_dmp_ground_primitive():
  303. assert dmp_ground_primitive([ZZ(1)], 0, ZZ) == (ZZ(1), [ZZ(1)])
  304. assert dmp_ground_primitive([[]], 1, ZZ) == (ZZ(0), [[]])
  305. assert dmp_ground_primitive(f_0, 2, ZZ) == (ZZ(1), f_0)
  306. assert dmp_ground_primitive(
  307. dmp_mul_ground(f_0, ZZ(2), 2, ZZ), 2, ZZ) == (ZZ(2), f_0)
  308. assert dmp_ground_primitive(f_1, 2, ZZ) == (ZZ(1), f_1)
  309. assert dmp_ground_primitive(
  310. dmp_mul_ground(f_1, ZZ(3), 2, ZZ), 2, ZZ) == (ZZ(3), f_1)
  311. assert dmp_ground_primitive(f_2, 2, ZZ) == (ZZ(1), f_2)
  312. assert dmp_ground_primitive(
  313. dmp_mul_ground(f_2, ZZ(4), 2, ZZ), 2, ZZ) == (ZZ(4), f_2)
  314. assert dmp_ground_primitive(f_3, 2, ZZ) == (ZZ(1), f_3)
  315. assert dmp_ground_primitive(
  316. dmp_mul_ground(f_3, ZZ(5), 2, ZZ), 2, ZZ) == (ZZ(5), f_3)
  317. assert dmp_ground_primitive(f_4, 2, ZZ) == (ZZ(1), f_4)
  318. assert dmp_ground_primitive(
  319. dmp_mul_ground(f_4, ZZ(6), 2, ZZ), 2, ZZ) == (ZZ(6), f_4)
  320. assert dmp_ground_primitive(f_5, 2, ZZ) == (ZZ(1), f_5)
  321. assert dmp_ground_primitive(
  322. dmp_mul_ground(f_5, ZZ(7), 2, ZZ), 2, ZZ) == (ZZ(7), f_5)
  323. assert dmp_ground_primitive(f_6, 3, ZZ) == (ZZ(1), f_6)
  324. assert dmp_ground_primitive(
  325. dmp_mul_ground(f_6, ZZ(8), 3, ZZ), 3, ZZ) == (ZZ(8), f_6)
  326. assert dmp_ground_primitive([[ZZ(2)]], 1, ZZ) == (ZZ(2), [[ZZ(1)]])
  327. assert dmp_ground_primitive([[QQ(2)]], 1, QQ) == (QQ(2), [[QQ(1)]])
  328. assert dmp_ground_primitive(
  329. [[QQ(2, 3)], [QQ(4, 9)]], 1, QQ) == (QQ(2, 9), [[QQ(3)], [QQ(2)]])
  330. assert dmp_ground_primitive(
  331. [[QQ(2, 3)], [QQ(4, 5)]], 1, QQ) == (QQ(2, 15), [[QQ(5)], [QQ(6)]])
  332. def test_dup_extract():
  333. f = dup_normal([2930944, 0, 2198208, 0, 549552, 0, 45796], ZZ)
  334. g = dup_normal([17585664, 0, 8792832, 0, 1099104, 0], ZZ)
  335. F = dup_normal([64, 0, 48, 0, 12, 0, 1], ZZ)
  336. G = dup_normal([384, 0, 192, 0, 24, 0], ZZ)
  337. assert dup_extract(f, g, ZZ) == (45796, F, G)
  338. def test_dmp_ground_extract():
  339. f = dmp_normal(
  340. [[2930944], [], [2198208], [], [549552], [], [45796]], 1, ZZ)
  341. g = dmp_normal([[17585664], [], [8792832], [], [1099104], []], 1, ZZ)
  342. F = dmp_normal([[64], [], [48], [], [12], [], [1]], 1, ZZ)
  343. G = dmp_normal([[384], [], [192], [], [24], []], 1, ZZ)
  344. assert dmp_ground_extract(f, g, 1, ZZ) == (45796, F, G)
  345. def test_dup_real_imag():
  346. assert dup_real_imag([], ZZ) == ([[]], [[]])
  347. assert dup_real_imag([1], ZZ) == ([[1]], [[]])
  348. assert dup_real_imag([1, 1], ZZ) == ([[1], [1]], [[1, 0]])
  349. assert dup_real_imag([1, 2], ZZ) == ([[1], [2]], [[1, 0]])
  350. assert dup_real_imag(
  351. [1, 2, 3], ZZ) == ([[1], [2], [-1, 0, 3]], [[2, 0], [2, 0]])
  352. assert dup_real_imag([ZZ(1), ZZ(0), ZZ(1), ZZ(3)], ZZ) == (
  353. [[ZZ(1)], [], [ZZ(-3), ZZ(0), ZZ(1)], [ZZ(3)]],
  354. [[ZZ(3), ZZ(0)], [], [ZZ(-1), ZZ(0), ZZ(1), ZZ(0)]]
  355. )
  356. raises(DomainError, lambda: dup_real_imag([EX(1), EX(2)], EX))
  357. def test_dup_mirror():
  358. assert dup_mirror([], ZZ) == []
  359. assert dup_mirror([1], ZZ) == [1]
  360. assert dup_mirror([1, 2, 3, 4, 5], ZZ) == [1, -2, 3, -4, 5]
  361. assert dup_mirror([1, 2, 3, 4, 5, 6], ZZ) == [-1, 2, -3, 4, -5, 6]
  362. def test_dup_scale():
  363. assert dup_scale([], -1, ZZ) == []
  364. assert dup_scale([1], -1, ZZ) == [1]
  365. assert dup_scale([1, 2, 3, 4, 5], -1, ZZ) == [1, -2, 3, -4, 5]
  366. assert dup_scale([1, 2, 3, 4, 5], -7, ZZ) == [2401, -686, 147, -28, 5]
  367. def test_dup_shift():
  368. assert dup_shift([], 1, ZZ) == []
  369. assert dup_shift([1], 1, ZZ) == [1]
  370. assert dup_shift([1, 2, 3, 4, 5], 1, ZZ) == [1, 6, 15, 20, 15]
  371. assert dup_shift([1, 2, 3, 4, 5], 7, ZZ) == [1, 30, 339, 1712, 3267]
  372. def test_dmp_shift():
  373. assert dmp_shift([ZZ(1), ZZ(2)], [ZZ(1)], 0, ZZ) == [ZZ(1), ZZ(3)]
  374. assert dmp_shift([[]], [ZZ(1), ZZ(2)], 1, ZZ) == [[]]
  375. xy = [[ZZ(1), ZZ(0)], []] # x*y
  376. x1y2 = [[ZZ(1), ZZ(2)], [ZZ(1), ZZ(2)]] # (x+1)*(y+2)
  377. assert dmp_shift(xy, [ZZ(1), ZZ(2)], 1, ZZ) == x1y2
  378. def test_dup_transform():
  379. assert dup_transform([], [], [1, 1], ZZ) == []
  380. assert dup_transform([], [1], [1, 1], ZZ) == []
  381. assert dup_transform([], [1, 2], [1, 1], ZZ) == []
  382. assert dup_transform([6, -5, 4, -3, 17], [1, -3, 4], [2, -3], ZZ) == \
  383. [6, -82, 541, -2205, 6277, -12723, 17191, -13603, 4773]
  384. def test_dup_compose():
  385. assert dup_compose([], [], ZZ) == []
  386. assert dup_compose([], [1], ZZ) == []
  387. assert dup_compose([], [1, 2], ZZ) == []
  388. assert dup_compose([1], [], ZZ) == [1]
  389. assert dup_compose([1, 2, 0], [], ZZ) == []
  390. assert dup_compose([1, 2, 1], [], ZZ) == [1]
  391. assert dup_compose([1, 2, 1], [1], ZZ) == [4]
  392. assert dup_compose([1, 2, 1], [7], ZZ) == [64]
  393. assert dup_compose([1, 2, 1], [1, -1], ZZ) == [1, 0, 0]
  394. assert dup_compose([1, 2, 1], [1, 1], ZZ) == [1, 4, 4]
  395. assert dup_compose([1, 2, 1], [1, 2, 1], ZZ) == [1, 4, 8, 8, 4]
  396. def test_dmp_compose():
  397. assert dmp_compose([1, 2, 1], [1, 2, 1], 0, ZZ) == [1, 4, 8, 8, 4]
  398. assert dmp_compose([[[]]], [[[]]], 2, ZZ) == [[[]]]
  399. assert dmp_compose([[[]]], [[[1]]], 2, ZZ) == [[[]]]
  400. assert dmp_compose([[[]]], [[[1]], [[2]]], 2, ZZ) == [[[]]]
  401. assert dmp_compose([[[1]]], [], 2, ZZ) == [[[1]]]
  402. assert dmp_compose([[1], [2], [ ]], [[]], 1, ZZ) == [[]]
  403. assert dmp_compose([[1], [2], [1]], [[]], 1, ZZ) == [[1]]
  404. assert dmp_compose([[1], [2], [1]], [[1]], 1, ZZ) == [[4]]
  405. assert dmp_compose([[1], [2], [1]], [[7]], 1, ZZ) == [[64]]
  406. assert dmp_compose([[1], [2], [1]], [[1], [-1]], 1, ZZ) == [[1], [ ], [ ]]
  407. assert dmp_compose([[1], [2], [1]], [[1], [ 1]], 1, ZZ) == [[1], [4], [4]]
  408. assert dmp_compose(
  409. [[1], [2], [1]], [[1], [2], [1]], 1, ZZ) == [[1], [4], [8], [8], [4]]
  410. def test_dup_decompose():
  411. assert dup_decompose([1], ZZ) == [[1]]
  412. assert dup_decompose([1, 0], ZZ) == [[1, 0]]
  413. assert dup_decompose([1, 0, 0, 0], ZZ) == [[1, 0, 0, 0]]
  414. assert dup_decompose([1, 0, 0, 0, 0], ZZ) == [[1, 0, 0], [1, 0, 0]]
  415. assert dup_decompose(
  416. [1, 0, 0, 0, 0, 0, 0], ZZ) == [[1, 0, 0, 0], [1, 0, 0]]
  417. assert dup_decompose([7, 0, 0, 0, 1], ZZ) == [[7, 0, 1], [1, 0, 0]]
  418. assert dup_decompose([4, 0, 3, 0, 2], ZZ) == [[4, 3, 2], [1, 0, 0]]
  419. f = [1, 0, 20, 0, 150, 0, 500, 0, 625, -2, 0, -10, 9]
  420. assert dup_decompose(f, ZZ) == [[1, 0, 0, -2, 9], [1, 0, 5, 0]]
  421. f = [2, 0, 40, 0, 300, 0, 1000, 0, 1250, -4, 0, -20, 18]
  422. assert dup_decompose(f, ZZ) == [[2, 0, 0, -4, 18], [1, 0, 5, 0]]
  423. f = [1, 0, 20, -8, 150, -120, 524, -600, 865, -1034, 600, -170, 29]
  424. assert dup_decompose(f, ZZ) == [[1, -8, 24, -34, 29], [1, 0, 5, 0]]
  425. R, t = ring("t", ZZ)
  426. f = [6*t**2 - 42,
  427. 48*t**2 + 96,
  428. 144*t**2 + 648*t + 288,
  429. 624*t**2 + 864*t + 384,
  430. 108*t**3 + 312*t**2 + 432*t + 192]
  431. assert dup_decompose(f, R.to_domain()) == [f]
  432. def test_dmp_lift():
  433. q = [QQ(1, 1), QQ(0, 1), QQ(1, 1)]
  434. f_a = [ANP([QQ(1, 1)], q, QQ), ANP([], q, QQ), ANP([], q, QQ),
  435. ANP([QQ(1, 1), QQ(0, 1)], q, QQ), ANP([QQ(17, 1), QQ(0, 1)], q, QQ)]
  436. f_lift = QQ.map([1, 0, 0, 0, 0, 0, 1, 34, 289])
  437. assert dmp_lift(f_a, 0, QQ.algebraic_field(I)) == f_lift
  438. f_g = [QQ_I(1), QQ_I(0), QQ_I(0), QQ_I(0, 1), QQ_I(0, 17)]
  439. assert dmp_lift(f_g, 0, QQ_I) == f_lift
  440. raises(DomainError, lambda: dmp_lift([EX(1), EX(2)], 0, EX))
  441. def test_dup_sign_variations():
  442. assert dup_sign_variations([], ZZ) == 0
  443. assert dup_sign_variations([1, 0], ZZ) == 0
  444. assert dup_sign_variations([1, 0, 2], ZZ) == 0
  445. assert dup_sign_variations([1, 0, 3, 0], ZZ) == 0
  446. assert dup_sign_variations([1, 0, 4, 0, 5], ZZ) == 0
  447. assert dup_sign_variations([-1, 0, 2], ZZ) == 1
  448. assert dup_sign_variations([-1, 0, 3, 0], ZZ) == 1
  449. assert dup_sign_variations([-1, 0, 4, 0, 5], ZZ) == 1
  450. assert dup_sign_variations([-1, -4, -5], ZZ) == 0
  451. assert dup_sign_variations([ 1, -4, -5], ZZ) == 1
  452. assert dup_sign_variations([ 1, 4, -5], ZZ) == 1
  453. assert dup_sign_variations([ 1, -4, 5], ZZ) == 2
  454. assert dup_sign_variations([-1, 4, -5], ZZ) == 2
  455. assert dup_sign_variations([-1, 4, 5], ZZ) == 1
  456. assert dup_sign_variations([-1, -4, 5], ZZ) == 1
  457. assert dup_sign_variations([ 1, 4, 5], ZZ) == 0
  458. assert dup_sign_variations([-1, 0, -4, 0, -5], ZZ) == 0
  459. assert dup_sign_variations([ 1, 0, -4, 0, -5], ZZ) == 1
  460. assert dup_sign_variations([ 1, 0, 4, 0, -5], ZZ) == 1
  461. assert dup_sign_variations([ 1, 0, -4, 0, 5], ZZ) == 2
  462. assert dup_sign_variations([-1, 0, 4, 0, -5], ZZ) == 2
  463. assert dup_sign_variations([-1, 0, 4, 0, 5], ZZ) == 1
  464. assert dup_sign_variations([-1, 0, -4, 0, 5], ZZ) == 1
  465. assert dup_sign_variations([ 1, 0, 4, 0, 5], ZZ) == 0
  466. def test_dup_clear_denoms():
  467. assert dup_clear_denoms([], QQ, ZZ) == (ZZ(1), [])
  468. assert dup_clear_denoms([QQ(1)], QQ, ZZ) == (ZZ(1), [QQ(1)])
  469. assert dup_clear_denoms([QQ(7)], QQ, ZZ) == (ZZ(1), [QQ(7)])
  470. assert dup_clear_denoms([QQ(7, 3)], QQ) == (ZZ(3), [QQ(7)])
  471. assert dup_clear_denoms([QQ(7, 3)], QQ, ZZ) == (ZZ(3), [QQ(7)])
  472. assert dup_clear_denoms(
  473. [QQ(3), QQ(1), QQ(0)], QQ, ZZ) == (ZZ(1), [QQ(3), QQ(1), QQ(0)])
  474. assert dup_clear_denoms(
  475. [QQ(1), QQ(1, 2), QQ(0)], QQ, ZZ) == (ZZ(2), [QQ(2), QQ(1), QQ(0)])
  476. assert dup_clear_denoms([QQ(3), QQ(
  477. 1), QQ(0)], QQ, ZZ, convert=True) == (ZZ(1), [ZZ(3), ZZ(1), ZZ(0)])
  478. assert dup_clear_denoms([QQ(1), QQ(
  479. 1, 2), QQ(0)], QQ, ZZ, convert=True) == (ZZ(2), [ZZ(2), ZZ(1), ZZ(0)])
  480. assert dup_clear_denoms(
  481. [EX(S(3)/2), EX(S(9)/4)], EX) == (EX(4), [EX(6), EX(9)])
  482. assert dup_clear_denoms([EX(7)], EX) == (EX(1), [EX(7)])
  483. assert dup_clear_denoms([EX(sin(x)/x), EX(0)], EX) == (EX(x), [EX(sin(x)), EX(0)])
  484. F = RR.frac_field(x)
  485. result = dup_clear_denoms([F(8.48717/(8.0089*x + 2.83)), F(0.0)], F)
  486. assert str(result) == "(x + 0.353356890459364, [1.05971731448763, 0.0])"
  487. def test_dmp_clear_denoms():
  488. assert dmp_clear_denoms([[]], 1, QQ, ZZ) == (ZZ(1), [[]])
  489. assert dmp_clear_denoms([[QQ(1)]], 1, QQ, ZZ) == (ZZ(1), [[QQ(1)]])
  490. assert dmp_clear_denoms([[QQ(7)]], 1, QQ, ZZ) == (ZZ(1), [[QQ(7)]])
  491. assert dmp_clear_denoms([[QQ(7, 3)]], 1, QQ) == (ZZ(3), [[QQ(7)]])
  492. assert dmp_clear_denoms([[QQ(7, 3)]], 1, QQ, ZZ) == (ZZ(3), [[QQ(7)]])
  493. assert dmp_clear_denoms(
  494. [[QQ(3)], [QQ(1)], []], 1, QQ, ZZ) == (ZZ(1), [[QQ(3)], [QQ(1)], []])
  495. assert dmp_clear_denoms([[QQ(
  496. 1)], [QQ(1, 2)], []], 1, QQ, ZZ) == (ZZ(2), [[QQ(2)], [QQ(1)], []])
  497. assert dmp_clear_denoms([QQ(3), QQ(
  498. 1), QQ(0)], 0, QQ, ZZ, convert=True) == (ZZ(1), [ZZ(3), ZZ(1), ZZ(0)])
  499. assert dmp_clear_denoms([QQ(1), QQ(1, 2), QQ(
  500. 0)], 0, QQ, ZZ, convert=True) == (ZZ(2), [ZZ(2), ZZ(1), ZZ(0)])
  501. assert dmp_clear_denoms([[QQ(3)], [QQ(
  502. 1)], []], 1, QQ, ZZ, convert=True) == (ZZ(1), [[QQ(3)], [QQ(1)], []])
  503. assert dmp_clear_denoms([[QQ(1)], [QQ(1, 2)], []], 1, QQ, ZZ,
  504. convert=True) == (ZZ(2), [[QQ(2)], [QQ(1)], []])
  505. assert dmp_clear_denoms(
  506. [[EX(S(3)/2)], [EX(S(9)/4)]], 1, EX) == (EX(4), [[EX(6)], [EX(9)]])
  507. assert dmp_clear_denoms([[EX(7)]], 1, EX) == (EX(1), [[EX(7)]])
  508. assert dmp_clear_denoms([[EX(sin(x)/x), EX(0)]], 1, EX) == (EX(x), [[EX(sin(x)), EX(0)]])