1pub use crate::report::SummaryReport;
12use crate::{PackageInfo, PackageMap, PackageStatus, Summary, SummaryId, SummarySource};
13use ahash::AHashMap;
14use diffus::{Diffable, edit};
15use semver::Version;
16use serde::{Serialize, ser::SerializeStruct};
17use std::{
18 collections::{BTreeMap, BTreeSet},
19 fmt, mem,
20};
21
22#[derive(Clone, Debug, Eq, PartialEq, Serialize)]
73#[serde(rename_all = "kebab-case")]
74pub struct SummaryDiff<'a> {
75 pub target_packages: PackageDiff<'a>,
77
78 pub host_packages: PackageDiff<'a>,
80}
81
82impl<'a> SummaryDiff<'a> {
83 pub fn new(old: &'a Summary, new: &'a Summary) -> Self {
85 Self {
86 target_packages: PackageDiff::new(&old.target_packages, &new.target_packages),
87 host_packages: PackageDiff::new(&old.host_packages, &new.host_packages),
88 }
89 }
90
91 pub fn is_changed(&self) -> bool {
93 !self.is_unchanged()
94 }
95
96 pub fn is_unchanged(&self) -> bool {
98 self.target_packages.is_unchanged() && self.host_packages.is_unchanged()
99 }
100
101 pub fn report<'b>(&'b self) -> SummaryReport<'a, 'b> {
105 SummaryReport::new(self)
106 }
107}
108
109pub type UnchangedInfo<'a> = (&'a Version, &'a SummarySource, &'a PackageInfo);
111
112#[derive(Clone, Debug, Eq, PartialEq)]
114pub struct PackageDiff<'a> {
115 pub changed: BTreeMap<&'a SummaryId, SummaryDiffStatus<'a>>,
117
118 pub unchanged: BTreeMap<&'a str, Vec<UnchangedInfo<'a>>>,
120}
121
122impl<'a> PackageDiff<'a> {
123 pub fn new(old: &'a PackageMap, new: &'a PackageMap) -> Self {
125 let mut changed = BTreeMap::new();
126 let mut unchanged = BTreeMap::new();
127
128 let mut add_unchanged = |summary_id: &'a SummaryId, info: &'a PackageInfo| {
129 unchanged
130 .entry(summary_id.name.as_str())
131 .or_insert_with(Vec::new)
132 .push((&summary_id.version, &summary_id.source, info));
133 };
134
135 match (*old).diff(new) {
136 edit::Edit::Copy(_) => {
137 for (summary_id, info) in new {
139 add_unchanged(summary_id, info);
140 }
141 }
142 edit::Edit::Change(diff) => {
143 for (summary_id, diff) in diff {
144 match diff {
145 edit::map::Edit::Copy(info) => {
146 add_unchanged(summary_id, info);
148 }
149 edit::map::Edit::Insert(info) => {
150 let status = SummaryDiffStatus::Added { info };
152 changed.insert(summary_id, status);
153 }
154 edit::map::Edit::Remove(old_info) => {
155 let status = SummaryDiffStatus::Removed { old_info };
157 changed.insert(summary_id, status);
158 }
159 edit::map::Edit::Change((old_info, new_info)) => {
160 let status =
162 SummaryDiffStatus::make_changed(None, None, old_info, new_info);
163 changed.insert(summary_id, status);
164 }
165 }
166 }
167 }
168 }
169
170 Self::combine_insert_remove(&mut changed);
172
173 Self { changed, unchanged }
174 }
175
176 pub fn is_unchanged(&self) -> bool {
178 self.changed.is_empty()
179 }
180
181 fn combine_insert_remove(changed: &mut BTreeMap<&'a SummaryId, SummaryDiffStatus<'a>>) {
186 let mut combine_statuses: AHashMap<&str, CombineStatus<'_>> =
187 AHashMap::with_capacity(changed.len());
188
189 for (summary_id, status) in &*changed {
190 let entry = combine_statuses
191 .entry(summary_id.name.as_str())
192 .or_insert_with(|| CombineStatus::None);
193 match status {
194 SummaryDiffStatus::Added { .. } => entry.record_added(summary_id),
195 SummaryDiffStatus::Removed { .. } => entry.record_removed(summary_id),
196 SummaryDiffStatus::Modified { .. } => entry.record_changed(),
197 }
198 }
199
200 for status in combine_statuses.values() {
201 if let CombineStatus::Combine { added, removed } = status {
202 let removed_status = changed
203 .remove(removed)
204 .expect("removed ID should be present");
205
206 let old_info = match removed_status {
207 SummaryDiffStatus::Removed { old_info } => old_info,
208 other => panic!("expected Removed, found {:?}", other),
209 };
210
211 let added_status = changed.get_mut(added).expect("added ID should be present");
212 let new_info = match &*added_status {
213 SummaryDiffStatus::Added { info } => *info,
214 other => panic!("expected Added, found {:?}", other),
215 };
216
217 let old_version = if added.version != removed.version {
218 Some(&removed.version)
219 } else {
220 None
221 };
222 let old_source = if added.source != removed.source {
223 Some(&removed.source)
224 } else {
225 None
226 };
227
228 let _ = mem::replace(
230 added_status,
231 SummaryDiffStatus::make_changed(old_version, old_source, old_info, new_info),
232 );
233 }
234 }
235 }
236}
237
238pub(crate) fn changed_sort_key<'a>(
239 summary_id: &'a SummaryId,
240 status: &SummaryDiffStatus<'_>,
241) -> impl Ord + 'a {
242 (status.tag(), status.latest_status(), summary_id)
248}
249
250impl Serialize for PackageDiff<'_> {
251 fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
252 where
253 S: serde::Serializer,
254 {
255 #[derive(Serialize)]
256 struct Changed<'a> {
257 #[serde(flatten)]
260 package: &'a SummaryId,
261 #[serde(flatten)]
262 changes: &'a SummaryDiffStatus<'a>,
263 }
264
265 let mut changed: Vec<Changed> = self
266 .changed
267 .iter()
268 .map(|(package, changes)| Changed { package, changes })
269 .collect();
270 changed.sort_by_key(|item| changed_sort_key(item.package, item.changes));
272
273 let mut state = serializer.serialize_struct("PackageDiff", 2)?;
274 state.serialize_field("changed", &changed)?;
275
276 #[derive(Serialize)]
277 struct Unchanged<'a> {
278 name: &'a str,
280 version: &'a Version,
281 #[serde(flatten)]
282 source: &'a SummarySource,
283 #[serde(flatten)]
284 info: &'a PackageInfo,
285 }
286
287 if !self.unchanged.is_empty() {
290 let mut unchanged: Vec<_> = self
291 .unchanged
292 .iter()
293 .flat_map(|(&name, info)| {
294 info.iter().map(move |(version, source, info)| Unchanged {
295 name,
296 version,
297 source,
298 info,
299 })
300 })
301 .collect();
302 unchanged.sort_by_key(|item| (item.name, item.version, item.source));
304 state.serialize_field("unchanged", &unchanged)?;
305 }
306
307 state.end()
308 }
309}
310
311#[derive(Clone, Debug, Eq, PartialEq, Serialize)]
313#[serde(rename_all = "kebab-case", tag = "change")]
314pub enum SummaryDiffStatus<'a> {
315 #[serde(rename_all = "kebab-case")]
317 Added {
318 #[serde(flatten)]
320 info: &'a PackageInfo,
321 },
322
323 #[serde(rename_all = "kebab-case")]
325 Removed {
326 #[serde(flatten, with = "removed_impl")]
328 old_info: &'a PackageInfo,
329 },
330
331 #[serde(rename_all = "kebab-case")]
335 Modified {
336 old_version: Option<&'a Version>,
338
339 old_source: Option<&'a SummarySource>,
341
342 old_status: Option<PackageStatus>,
344
345 new_status: PackageStatus,
347
348 added_features: BTreeSet<&'a str>,
350
351 removed_features: BTreeSet<&'a str>,
353
354 unchanged_features: BTreeSet<&'a str>,
356
357 #[serde(default)]
359 added_optional_deps: BTreeSet<&'a str>,
360
361 #[serde(default)]
363 removed_optional_deps: BTreeSet<&'a str>,
364
365 #[serde(default)]
367 unchanged_optional_deps: BTreeSet<&'a str>,
368 },
369}
370
371impl<'a> SummaryDiffStatus<'a> {
372 fn make_changed(
373 old_version: Option<&'a Version>,
374 old_source: Option<&'a SummarySource>,
375 old_info: &'a PackageInfo,
376 new_info: &'a PackageInfo,
377 ) -> Self {
378 let old_status = if old_info.status != new_info.status {
379 Some(old_info.status)
380 } else {
381 None
382 };
383
384 let [added_features, removed_features, unchanged_features] =
385 Self::make_changed_diff(&old_info.features, &new_info.features);
386
387 let [
388 added_optional_deps,
389 removed_optional_deps,
390 unchanged_optional_deps,
391 ] = Self::make_changed_diff(&old_info.optional_deps, &new_info.optional_deps);
392
393 SummaryDiffStatus::Modified {
394 old_version,
395 old_source,
396 old_status,
397 new_status: new_info.status,
398 added_features,
399 removed_features,
400 unchanged_features,
401 added_optional_deps,
402 removed_optional_deps,
403 unchanged_optional_deps,
404 }
405 }
406
407 fn make_changed_diff(
408 old_features: &'a BTreeSet<String>,
409 new_features: &'a BTreeSet<String>,
410 ) -> [BTreeSet<&'a str>; 3] {
411 let mut added_features = BTreeSet::new();
412 let mut removed_features = BTreeSet::new();
413 let mut unchanged_features = BTreeSet::new();
414
415 match old_features.diff(new_features) {
416 edit::Edit::Copy(features) => {
417 unchanged_features.extend(features.iter().map(|feature| feature.as_str()));
418 }
419 edit::Edit::Change(diff) => {
420 for (_, diff) in diff {
421 match diff {
422 edit::set::Edit::Copy(feature) => {
423 unchanged_features.insert(feature.as_str());
424 }
425 edit::set::Edit::Insert(feature) => {
426 added_features.insert(feature.as_str());
427 }
428 edit::set::Edit::Remove(feature) => {
429 removed_features.insert(feature.as_str());
430 }
431 }
432 }
433 }
434 }
435
436 [added_features, removed_features, unchanged_features]
437 }
438
439 pub fn tag(&self) -> SummaryDiffTag {
443 match self {
444 SummaryDiffStatus::Added { .. } => SummaryDiffTag::Added,
445 SummaryDiffStatus::Removed { .. } => SummaryDiffTag::Removed,
446 SummaryDiffStatus::Modified { .. } => SummaryDiffTag::Modified,
447 }
448 }
449
450 pub fn latest_status(&self) -> PackageStatus {
452 match self {
453 SummaryDiffStatus::Added { info } => info.status,
454 SummaryDiffStatus::Removed { old_info } => old_info.status,
455 SummaryDiffStatus::Modified { new_status, .. } => *new_status,
456 }
457 }
458}
459
460mod removed_impl {
461 use super::*;
462 use serde::Serializer;
463
464 pub fn serialize<S>(item: &PackageInfo, serializer: S) -> Result<S::Ok, S::Error>
465 where
466 S: Serializer,
467 {
468 #[derive(Serialize)]
469 #[serde(rename_all = "kebab-case")]
470 struct OldPackageInfo<'a> {
471 old_status: &'a PackageStatus,
472 old_features: &'a BTreeSet<String>,
473 }
474
475 let old_info = OldPackageInfo {
476 old_status: &item.status,
477 old_features: &item.features,
478 };
479
480 old_info.serialize(serializer)
481 }
482}
483
484#[derive(Copy, Clone, Debug, Eq, Ord, PartialEq, PartialOrd)]
488pub enum SummaryDiffTag {
489 Added,
491
492 Modified,
494
495 Removed,
497}
498
499impl fmt::Display for SummaryDiffTag {
500 fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
501 match self {
502 SummaryDiffTag::Added => write!(f, "A"),
503 SummaryDiffTag::Modified => write!(f, "M"),
504 SummaryDiffTag::Removed => write!(f, "R"),
505 }
506 }
507}
508
509impl<'a> Diffable<'a> for PackageInfo {
510 type Diff = (&'a PackageInfo, &'a PackageInfo);
511
512 fn diff(&'a self, other: &'a Self) -> edit::Edit<'a, Self> {
513 if self == other {
514 edit::Edit::Copy(self)
515 } else {
516 edit::Edit::Change((self, other))
517 }
518 }
519}
520
521impl<'a> Diffable<'a> for PackageStatus {
522 type Diff = (&'a PackageStatus, &'a PackageStatus);
523
524 fn diff(&'a self, other: &'a Self) -> edit::Edit<'a, Self> {
525 if self == other {
526 edit::Edit::Copy(self)
527 } else {
528 edit::Edit::Change((self, other))
529 }
530 }
531}
532
533enum CombineStatus<'a> {
535 None,
536 Added(&'a SummaryId),
537 Removed(&'a SummaryId),
538 Combine {
539 added: &'a SummaryId,
540 removed: &'a SummaryId,
541 },
542 Ignore,
543}
544
545impl<'a> CombineStatus<'a> {
546 fn record_added(&mut self, summary_id: &'a SummaryId) {
547 let new = match self {
548 CombineStatus::None => CombineStatus::Added(summary_id),
549 CombineStatus::Removed(removed) => CombineStatus::Combine {
550 added: summary_id,
551 removed,
552 },
553 _ => CombineStatus::Ignore,
554 };
555
556 let _ = mem::replace(self, new);
557 }
558
559 fn record_removed(&mut self, summary_id: &'a SummaryId) {
560 let new = match self {
561 CombineStatus::None => CombineStatus::Removed(summary_id),
562 CombineStatus::Added(added) => CombineStatus::Combine {
563 added,
564 removed: summary_id,
565 },
566 _ => CombineStatus::Ignore,
567 };
568
569 let _ = mem::replace(self, new);
570 }
571
572 fn record_changed(&mut self) {
573 let _ = mem::replace(self, CombineStatus::Ignore);
576 }
577}