Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add prime factorization and its properties #1969

Merged
merged 46 commits into from
Mar 16, 2024
Merged
Show file tree
Hide file tree
Changes from 20 commits
Commits
Show all changes
46 commits
Select commit Hold shift + click to select a range
98dff4f
Add prime factorization and its properties
Taneb May 13, 2023
54ed74c
Add missing header comment
Taneb May 13, 2023
9c05449
Remove completely trivial lemma
Taneb May 15, 2023
6c9a9e1
Use equational reasoning in quotient|n proof
Taneb May 15, 2023
7ec3801
Fix typo in module header
Taneb May 15, 2023
8e0d749
Factorization => Factorisation
Taneb May 15, 2023
f77023e
Use Nat lemma in extend-|/
Taneb May 15, 2023
2f4193c
[ cleanup ] part of the proof
gallais May 15, 2023
1b45ff5
[ cleanup ] finishing up the job
gallais May 15, 2023
cec5fac
[ cleanup ] a little bit more
gallais May 15, 2023
4c8b43e
[ cleanup ] the import list
gallais May 15, 2023
dd898f9
[ fix ] header style
gallais May 15, 2023
b330ae5
Merge branch 'master' into prime-factorization
gallais May 15, 2023
bec7b21
[ fix ] broken merge: missing import
gallais May 15, 2023
294bdad
Move Data.Nat.Rough to Data.Nat.Primality.Rough
Taneb Jun 12, 2023
8796eee
Rename productPreserves↭⇒≡ to product-↭
Taneb Jun 12, 2023
f0948e8
Use proof of Prime=>NonZero
Taneb Jun 12, 2023
2a1ab80
Open reasoning once in factoriseRec
Taneb Jun 12, 2023
7e73c8d
Rename Factorisation => PrimeFactorisation
Taneb Jun 12, 2023
c2a712f
Move wheres around
Taneb Jun 12, 2023
6262285
Merge branch 'master' into prime-factorization
Taneb Aug 15, 2023
4073e5f
Tidy up Rough a bit
Taneb Aug 15, 2023
8afce74
Move quotient|n to top of file
Taneb Aug 15, 2023
87022d8
Replace factorisationPullToFront with slightly more generally useful …
Taneb Aug 15, 2023
ea1142d
Merge branch 'master' into prime-factorization
Taneb Aug 21, 2023
58a71c5
Merge branch 'master' into prime-factorization
Taneb Oct 3, 2023
230d498
Fix import after merge
Taneb Oct 3, 2023
1f1ed8e
Clean up proof of 2-rough-n
Taneb Oct 3, 2023
8b34d27
Make argument to 2-rough-n implicit
Taneb Oct 3, 2023
7482c46
Rename 2-rough-n to 2-rough
Taneb Oct 3, 2023
20d8d66
Merge branch 'master' into prime-factorization
Taneb Dec 21, 2023
aca45ad
Complete merge, rewrite factorisation logic a bit
Taneb Dec 21, 2023
e24f0be
Short circuit when candidate is greater than square root of product
Taneb Dec 22, 2023
fd4e806
Remove redefined lemma
Taneb Dec 22, 2023
510b7ec
Minor simplifications
Taneb Dec 22, 2023
9f2ef51
Remove private pattern synonyms
Taneb Dec 22, 2023
217d0cd
Change name of lemma
Taneb Jan 31, 2024
286bc7e
Typo
Taneb Jan 31, 2024
5063190
Remove using list from import
Taneb Jan 31, 2024
cf8a9c4
Clean up proof
Taneb Jan 31, 2024
35153ca
Merge remote-tracking branch 'origin/master' into prime-factorization
Taneb Feb 1, 2024
578718d
Fixes after merge
Taneb Feb 1, 2024
a556522
Addressed some feedback
MatthewDaggitt Mar 3, 2024
cc8cef2
Merge branch 'master' into prime-factorization
MatthewDaggitt Mar 3, 2024
7dcd066
Merge branch 'master' into prime-factorization
MatthewDaggitt Mar 16, 2024
eb893ed
Fix previous merge
MatthewDaggitt Mar 16, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,8 @@ open import Algebra.Bundles
open import Algebra.Definitions
open import Algebra.Structures
open import Data.Bool.Base using (Bool; true; false)
open import Data.Nat using (suc)
open import Data.Nat.Base using (suc; _*_)
open import Data.Nat.Properties using (*-assoc; *-comm)
open import Data.Product using (-,_; proj₂)
open import Data.List.Base as List
open import Data.List.Relation.Binary.Permutation.Propositional
Expand All @@ -29,11 +30,9 @@ open import Level using (Level)
open import Relation.Unary using (Pred)
open import Relation.Binary
open import Relation.Binary.PropositionalEquality as ≡
using (_≡_ ; refl ; cong; cong₂; _≢_)
using (_≡_ ; refl ; cong; cong₂; _≢_; module ≡-Reasoning)
open import Relation.Nullary

open PermutationReasoning

private
variable
a b p : Level
Expand Down Expand Up @@ -173,6 +172,7 @@ shift v (x ∷ xs) ys = begin
x ∷ (xs ++ [ v ] ++ ys) <⟨ shift v xs ys ⟩
x ∷ v ∷ xs ++ ys <<⟨ refl ⟩
v ∷ x ∷ xs ++ ys ∎
where open PermutationReasoning

drop-mid-≡ : ∀ {x : A} ws xs {ys} {zs} →
ws ++ [ x ] ++ ys ≡ xs ++ [ x ] ++ zs →
Expand Down Expand Up @@ -217,11 +217,13 @@ drop-mid {A = A} {x} ws xs p = drop-mid′ p ws xs refl refl
_ ∷ (xs ++ _ ∷ _) <⟨ shift _ _ _ ⟩
_ ∷ _ ∷ xs ++ _ <<⟨ refl ⟩
_ ∷ _ ∷ xs ++ _ ∎
where open PermutationReasoning
drop-mid′ (swap y z p) (y ∷ z ∷ ws) (z ∷ []) refl refl = begin
_ ∷ _ ∷ ws ++ _ <<⟨ refl ⟩
_ ∷ (_ ∷ ws ++ _) <⟨ ↭-sym (shift _ _ _) ⟩
_ ∷ (ws ++ _ ∷ _) <⟨ p ⟩
_ ∷ _ ∎
where open PermutationReasoning
drop-mid′ (swap y z p) (y ∷ z ∷ ws) (z ∷ y ∷ xs) refl refl = swap y z (drop-mid′ p _ _ refl refl)
drop-mid′ (trans p₁ p₂) ws xs refl refl with ∈-∃++ (∈-resp-↭ p₁ (∈-insert ws))
... | (h , t , refl) = trans (drop-mid′ p₁ ws h refl refl) (drop-mid′ p₂ h xs refl refl)
Expand All @@ -246,6 +248,7 @@ drop-mid {A = A} {x} ws xs p = drop-mid′ p ws xs refl refl
x ∷ xs ++ ys <⟨ ++-comm xs ys ⟩
x ∷ ys ++ xs ↭˘⟨ shift x ys xs ⟩
ys ++ (x ∷ xs) ∎
where open PermutationReasoning

++-isMagma : IsMagma {A = List A} _↭_ _++_
++-isMagma = record
Expand Down Expand Up @@ -301,6 +304,7 @@ shifts xs ys {zs} = begin
(xs ++ ys) ++ zs ↭⟨ ++⁺ʳ zs (++-comm xs ys) ⟩
(ys ++ xs) ++ zs ↭⟨ ++-assoc ys xs zs ⟩
ys ++ xs ++ zs ∎
where open PermutationReasoning

------------------------------------------------------------------------
-- _∷_
Expand All @@ -316,6 +320,7 @@ drop-∷ = drop-mid [] []
xs ++ [ x ] ↭⟨ shift x xs [] ⟩
x ∷ xs ++ [] ≡⟨ Lₚ.++-identityʳ _ ⟩
x ∷ xs ∎)
where open PermutationReasoning

------------------------------------------------------------------------
-- ʳ++
Expand Down Expand Up @@ -354,3 +359,17 @@ module _ {ℓ} {R : Rel A ℓ} (R? : Decidable R) where
(x ∷ xs) ++ y ∷ ys ≡˘⟨ Lₚ.++-assoc [ x ] xs (y ∷ ys) ⟩
x ∷ xs ++ y ∷ ys ∎
where open PermutationReasoning

------------------------------------------------------------------------
-- product

product-↭ : product Preserves _↭_ ⟶ _≡_
product-↭ refl = refl
product-↭ (prep x r) = cong (x *_) (product-↭ r)
product-↭ (trans r s) = ≡.trans (product-↭ r) (product-↭ s)
product-↭ (swap {xs} {ys} x y r) = begin
x * (y * product xs) ≡˘⟨ *-assoc x y (product xs) ⟩
(x * y) * product xs ≡⟨ cong₂ _*_ (*-comm x y) (product-↭ r) ⟩
(y * x) * product ys ≡⟨ *-assoc y x (product ys) ⟩
y * (x * product ys) ∎
where open ≡-Reasoning
10 changes: 10 additions & 0 deletions src/Data/Nat/Divisibility.agda
Original file line number Diff line number Diff line change
Expand Up @@ -300,3 +300,13 @@ m≤n⇒m!∣n! m≤n = help (≤⇒≤′ m≤n)
help : ∀ {m n} → m ≤′ n → m ! ∣ n !
help {m} {n} ≤′-refl = ∣-refl
help {m} {suc n} (≤′-step m≤′n) = ∣n⇒∣m*n (suc n) (help m≤′n)

------------------------------------------------------------------------
-- Properties of quotient

quotient∣n : ∀ {m n} (m∣n : m ∣ n) → quotient m∣n ∣ n
quotient∣n {m = m} {n = n} m∣n = divides m $ begin-equality
Taneb marked this conversation as resolved.
Show resolved Hide resolved
n ≡⟨ _∣_.equality m∣n ⟩
quotient m∣n * m ≡⟨ *-comm (quotient m∣n) m ⟩
m * quotient m∣n ∎
where open ≤-Reasoning
19 changes: 17 additions & 2 deletions src/Data/Nat/Primality.agda
Original file line number Diff line number Diff line change
Expand Up @@ -20,8 +20,8 @@ open import Relation.Nullary.Decidable as Dec
using (yes; no; from-yes; ¬?; decidable-stable; _×-dec_; _→-dec_)
open import Relation.Nullary.Negation using (¬_; contradiction)
open import Relation.Unary using (Decidable)
open import Relation.Binary.PropositionalEquality
using (refl; sym; cong; subst)
open import Relation.Binary.PropositionalEquality.Core
using (_≡_; refl; sym; cong)

private
variable
Expand Down Expand Up @@ -141,3 +141,18 @@ private
-- Example: 6 is composite
6-is-composite : Composite 6
6-is-composite = from-yes (composite? 6)

------------------------------------------------------------------------
-- Other properties

Prime⇒NonZero : Prime n → NonZero n
Prime⇒NonZero {suc _} _ = _

Composite⇒NonZero : Composite n → NonZero n
Composite⇒NonZero {suc _} _ = _

-- If m is a factor of prime p, then it is equal to either 1 or p itself
∣p⇒≡1∨≡p : ∀ m {p} → Prime p → m ∣ p → m ≡ 1 ⊎ m ≡ p
∣p⇒≡1∨≡p 0 {p} p-prime m∣p = contradiction (0∣⇒≡0 m∣p) (≢-nonZero⁻¹ p {{Prime⇒NonZero p-prime}})
∣p⇒≡1∨≡p 1 {p} p-prime m∣p = inj₁ refl
∣p⇒≡1∨≡p (suc (suc m)) {suc (suc p)} p-prime m∣p = inj₂ (≤∧≮⇒≡ (∣⇒≤ m∣p) λ m<p → p-prime (s≤s (s≤s z≤n)) m<p m∣p)
Taneb marked this conversation as resolved.
Show resolved Hide resolved
218 changes: 218 additions & 0 deletions src/Data/Nat/Primality/Factorisation.agda
Original file line number Diff line number Diff line change
@@ -0,0 +1,218 @@
------------------------------------------------------------------------
-- The Agda standard library
--
-- Prime factorisation of natural numbers and its properties
------------------------------------------------------------------------

{-# OPTIONS --cubical-compatible --safe #-}

module Data.Nat.Primality.Factorisation where

open import Data.Empty using (⊥-elim)
open import Data.Nat.Base
open import Data.Nat.Divisibility using (_∣_; _∣?_; quotient; quotient∣n; ∣-trans; ∣1⇒≡1; divides)
open import Data.Nat.Properties
open import Data.Nat.Induction using (<-Rec; <-rec)
open import Data.Nat.Primality using (Prime; euclidsLemma; ∣p⇒≡1∨≡p; Prime⇒NonZero)
open import Data.Nat.Primality.Rough using (_Rough_; 2-rough-n; extend-∤; roughn∧∣n⇒prime)
open import Data.Product as Π using (∃-syntax; _,_; proj₁; proj₂)
open import Data.List.Base using (List; []; _∷_; product)
open import Data.List.Relation.Unary.All as All using (All; []; _∷_)
open import Data.List.Relation.Binary.Permutation.Propositional as ↭
using (_↭_; prep; swap; ↭-refl; refl; module PermutationReasoning)
open import Data.List.Relation.Binary.Permutation.Propositional.Properties using (product-↭; All-resp-↭)
open import Data.Sum.Base using (inj₁; inj₂)
open import Function.Base using (_$_; _∘_; _|>_; flip)
open import Relation.Nullary.Decidable using (yes; no)
open import Relation.Nullary.Negation using (contradiction)
open import Relation.Binary.PropositionalEquality using (_≡_; refl; cong; module ≡-Reasoning)

------------------------------------------------------------------------
-- Core definition

record PrimeFactorisation (n : ℕ) : Set where
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The 'traditional' notion of PrimeFactorisation uses a list of (multiplicity, prime) pairs and has the extra invariant that all primes are distinct.

I am not suggesting you change this definition - but you should probably comment on your design choice, so that this aspect doesn't puzzle a maintainer years down the road.

field
factors : List ℕ
isFactorisation : product factors ≡ n
MatthewDaggitt marked this conversation as resolved.
Show resolved Hide resolved
factorsPrime : All Prime factors

MatthewDaggitt marked this conversation as resolved.
Show resolved Hide resolved
open PrimeFactorisation public using (factors)
open PrimeFactorisation

------------------------------------------------------------------------
-- Finding a factorisation
Taneb marked this conversation as resolved.
Show resolved Hide resolved

-- this builds up three important things:
-- * a proof that every number we've gotten to so far has increasingly higher
-- possible least prime factor, so we don't have to repeat smaller factores
MatthewDaggitt marked this conversation as resolved.
Show resolved Hide resolved
-- over and over (this is the "k" and "k-rough-n" parameters)
-- * a witness that this limit is getting closer to the number of interest, in a
-- way that helps the termination checker (the k≤n parameter)
-- * a proof that we can factorise any smaller number, which is useful when we
-- encounter a factor, as we can then divide by that factor and continue from
-- there without termination issues

private
pattern 2+ n = suc (suc n)
pattern 2≤2+n = s≤s (s≤s z≤n)
pattern 1<2+n = 2≤2+n
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Apologies for insisting on having removed such things from earlier versions of the library. Do we need to reinstate them full-bloodedly in Data.Nat.Base?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think these were snuck in by @gallais here so I'm going to refer this to him

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK, and definitionally equal, but notationally distinct:

pattern 1<2+n = s<s z<s

(I love pattern synonyms, too much perhaps)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's now the case that we have in Data.Nat.Base:

  • 2+ as a pattern synonym
  • sz<ss for the (second and) third item

So I think these can all be removed from here?


factorise : ∀ n → .{{NonZero n}} → PrimeFactorisation n
factorise 1 = record
{ factors = []
; isFactorisation = refl
; factorsPrime = []
}
factorise (2+ n) = <-rec P factoriseRec (2 + n) {2} 2≤2+n (≤⇒≤‴ 2≤2+n) (2-rough-n (2 + n))
where

P : ℕ → Set
P n′ = ∀ {k} → 2 ≤ n′ → k ≤‴ n′ → k Rough n′ → PrimeFactorisation n′

factoriseRec : ∀ n → <-Rec P n → P n
factoriseRec (2+ n) rec (s≤s (s≤s n≤z)) ≤‴-refl k-rough-n = record
{ factors = 2 + n ∷ []
; isFactorisation = *-identityʳ (2 + n)
Copy link
Contributor

@jamesmckinna jamesmckinna Oct 24, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well, perhaps I am mistaken here? Needs sym on my formulation. Ah well.

; factorsPrime = k-rough-n ∷ []
}
factoriseRec (2+ n) rec {0} 2≤2+n (≤‴-step (≤‴-step k<n)) k-rough-n =
factoriseRec (2+ n) rec 2≤2+n k<n (2-rough-n (2+ n))
factoriseRec (2+ n) rec {1} 2≤2+n (≤‴-step k<n) k-rough-n =
factoriseRec (2+ n) rec 2≤2+n k<n (2-rough-n (2+ n))
factoriseRec (2+ n) rec {suc (suc k)} 2≤2+n (≤‴-step k<n) k-rough-n with 2 + k ∣? 2+ n
... | no k∤n = factoriseRec (2+ n) rec {3 + k} 2≤2+n k<n (extend-∤ k-rough-n k∤n)
... | yes k∣n = record
{ factors = 2 + k ∷ factors res
; isFactorisation = prop
; factorsPrime = roughn∧∣n⇒prime k-rough-n 2≤2+n k∣n ∷ factorsPrime res
}
where
open ≤-Reasoning

q : ℕ
q = quotient k∣n

-- we know that k < n, so if q is 0 or 1 then q * k < n
2≤q : 2 ≤ q
2≤q = ≮⇒≥ (λ q<2 → contradiction (_∣_.equality k∣n) (>⇒≢ (prf (≤-pred q<2)))) where
Taneb marked this conversation as resolved.
Show resolved Hide resolved

prf : q ≤ 1 → q * 2+ k < 2 + n
prf q≤1 = begin-strict
q * 2+ k ≤⟨ *-monoˡ-≤ (2 + k) q≤1 ⟩
2 + k + 0 ≡⟨ +-identityʳ (2 + k) ⟩
2 + k <⟨ ≤‴⇒≤ k<n ⟩
2 + n ∎

0<q : 0 < q
0<q = begin-strict
0 <⟨ s≤s z≤n ⟩
2 ≤⟨ 2≤q ⟩
q ∎

q<n : q < 2 + n
q<n = begin-strict
q <⟨ m<m*n q (2 + k) ⦃ >-nonZero 0<q ⦄ 2≤2+n ⟩
q * (2 + k) ≡˘⟨ _∣_.equality k∣n ⟩
2 + n ∎

q≮2+k : q ≮ 2 + k
q≮2+k q<k = k-rough-n 2≤q q<k (quotient∣n k∣n)

res : PrimeFactorisation q
res = rec q q<n {2 + k} 2≤q (≤⇒≤‴ (≮⇒≥ q≮2+k))
$ λ {d} d<k d-prime → k-rough-n d<k d-prime ∘ flip ∣-trans (quotient∣n k∣n)

prop : (2 + k) * product (factors res) ≡ 2 + n
prop = begin-equality
(2 + k) * product (factors res)
≡⟨ cong ((2 + k) *_) (isFactorisation res) ⟩
(2 + k) * q
≡⟨ *-comm (2 + k) q ⟩
q * (2 + k)
≡˘⟨ _∣_.equality k∣n ⟩
2 + n ∎

------------------------------------------------------------------------
-- Properties of a factorisation

factorisation≥1 : ∀ {as} → All Prime as → product as ≥ 1
MatthewDaggitt marked this conversation as resolved.
Show resolved Hide resolved
factorisation≥1 {[]} [] = s≤s z≤n
factorisation≥1 {suc a ∷ as} (pa ∷ asPrime) = *-mono-≤ {1} {1 + a} (s≤s z≤n) (factorisation≥1 asPrime)
MatthewDaggitt marked this conversation as resolved.
Show resolved Hide resolved

factorisationPullToFront : ∀ {as} {p} → Prime p → p ∣ product as → All Prime as → ∃[ as′ ] as ↭ (p ∷ as′)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wouldn't it be 'simpler' if this took a PrimeFactorisation as input instead of the pieces it contains?

Also it feels like this could be proven more simply:

  1. prove p is a member of the list as
  2. then there's a permutation that puts it at the front.

This would be straightforward if you already knew it was a PrimeFactorisation.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've simplified it a bit. I don't think I do want it to take a PrimeFactorisation, because it recurses on the list, which would be a little more awkward. But maybe I'm wrong here, I'll think about it more

factorisationPullToFront {[]} {suc (suc p)} pPrime p∣Πas asPrime = contradiction (∣1⇒≡1 p∣Πas) λ ()
factorisationPullToFront {a ∷ as} {p} pPrime p∣aΠas (aPrime ∷ asPrime)
with euclidsLemma a (product as) pPrime p∣aΠas
... | inj₂ p∣Πas = Π.map (a ∷_) step ih
where
ih : ∃[ as′ ] as ↭ (p ∷ as′)
ih = factorisationPullToFront pPrime p∣Πas asPrime

step : ∀ {as′} → as ↭ p ∷ as′ → a ∷ as ↭ p ∷ a ∷ as′
step {as′} as↭p∷as′ = begin
a ∷ as ↭⟨ prep a as↭p∷as′ ⟩
a ∷ p ∷ as′ ↭⟨ swap a p refl ⟩
p ∷ a ∷ as′ ∎
where open PermutationReasoning

... | inj₁ p∣a with ∣p⇒≡1∨≡p p aPrime p∣a
... | inj₁ refl = ⊥-elim pPrime
... | inj₂ refl = as , ↭-refl

factorisationUnique′ : (as bs : List ℕ) → product as ≡ product bs → All Prime as → All Prime bs → as ↭ bs
MatthewDaggitt marked this conversation as resolved.
Show resolved Hide resolved
factorisationUnique′ [] [] Πas≡Πbs asPrime bsPrime = refl
JacquesCarette marked this conversation as resolved.
Show resolved Hide resolved
factorisationUnique′ [] (suc (suc b) ∷ bs) Πas≡Πbs asPrime (bPrime ∷ bsPrime) =
contradiction Πas≡Πbs (<⇒≢ Πas<Πbs)
where
Πas<Πbs : product [] < product (2+ b ∷ bs)
Πas<Πbs = begin-strict
1 ≡⟨⟩
1 * 1 <⟨ *-monoˡ-< 1 {1} {2 + b} 1<2+n ⟩
(2 + b) * 1 ≤⟨ *-monoʳ-≤ (2 + b) (factorisation≥1 bsPrime) ⟩
2+ b * product bs ≡⟨⟩
product (2+ b ∷ bs) ∎
jamesmckinna marked this conversation as resolved.
Show resolved Hide resolved
where open ≤-Reasoning

factorisationUnique′ (a ∷ as) bs Πas≡Πbs (aPrime ∷ asPrime) bsPrime = a∷as↭bs
where
a∣Πbs : a ∣ product bs
a∣Πbs = divides (product as) $ begin
product bs ≡˘⟨ Πas≡Πbs ⟩
product (a ∷ as) ≡⟨⟩
a * product as ≡⟨ *-comm a (product as) ⟩
product as * a ∎
where open ≡-Reasoning

shuffle : ∃[ bs′ ] bs ↭ a ∷ bs′
shuffle = factorisationPullToFront aPrime a∣Πbs bsPrime

bs′ = proj₁ shuffle
bs↭a∷bs′ = proj₂ shuffle

Πas≡Πbs′ : product as ≡ product bs′
Πas≡Πbs′ = *-cancelˡ-≡ (product as) (product bs′) a {{Prime⇒NonZero aPrime}} $ begin
a * product as ≡⟨ Πas≡Πbs ⟩
product bs ≡⟨ product-↭ bs↭a∷bs′ ⟩
a * product bs′ ∎
where open ≡-Reasoning

bs′Prime : All Prime bs′
bs′Prime = All.tail (All-resp-↭ bs↭a∷bs′ bsPrime)

a∷as↭bs : a ∷ as ↭ bs
a∷as↭bs = begin
a ∷ as <⟨ factorisationUnique′ as bs′ Πas≡Πbs′ asPrime bs′Prime ⟩
a ∷ bs′ ↭˘⟨ bs↭a∷bs′ ⟩
bs ∎
where open PermutationReasoning

factorisationUnique : {n : ℕ} (f f′ : PrimeFactorisation n) → factors f ↭ factors f′
factorisationUnique {n} f f′ =
factorisationUnique′ (factors f) (factors f′) Πf≡Πf′ (factorsPrime f) (factorsPrime f′)
where
Πf≡Πf′ : product (factors f) ≡ product (factors f′)
Πf≡Πf′ = begin
product (factors f) ≡⟨ isFactorisation f ⟩
n ≡˘⟨ isFactorisation f′ ⟩
product (factors f′) ∎
where open ≡-Reasoning
Loading