Compare commits
1 Commits
2bead6af27
...
3b26ff9fc1
| Author | SHA1 | Date |
|---|---|---|
|
|
3b26ff9fc1 | 3 weeks ago |
6 changed files with 196 additions and 0 deletions
@ -0,0 +1,2 @@
|
||||
schema: spec-driven |
||||
created: 2026-04-13 |
||||
@ -0,0 +1,67 @@
|
||||
## Context |
||||
|
||||
Current implementation in `pkg/pacman/pacman.go` spawns shell processes for pacman queries: |
||||
|
||||
1. **`ValidatePackage`**: Calls `pacman -Qip` then `pacman -Sip` (2 processes per package) to check if package exists in sync databases |
||||
2. **`MarkExplicit`**: Calls `pacman -D --explicit <pkg>` individually per package (N processes for N packages) |
||||
3. **AUR**: Called individually per package not found in pacman databases |
||||
|
||||
Performance issues scale with package count. No caching of package queries across calls. |
||||
|
||||
## Goals / Non-Goals |
||||
|
||||
**Goals:** |
||||
- Use Jguer/dyalpm library to query pacman databases without spawning processes |
||||
- Add in-memory cache for ALL package query results (pacman + AUR), valid for entire job duration |
||||
- Batch `pacman -D --explicit` calls to single process for multiple packages |
||||
- Batch AUR HTTP queries to single request for all packages not found in pacman |
||||
|
||||
**Non-Goals:** |
||||
- Refactor AUR handling (already uses HTTP API - will batch it) |
||||
- Add persistent cache (only job-duration in-memory) |
||||
- Change other pacman operations (sync, cleanup) |
||||
|
||||
## Decisions |
||||
|
||||
1. **dyalpm over go-alpm**: dyalpm uses purego (no cgo), cleaner cross-compilation |
||||
- Alternative: go-alpm (cgo-based) - rejected for compilation complexity |
||||
|
||||
2. **Unified cache in Pac struct**: Single cache map replaces separate aurCache |
||||
- Alternative: keep separate caches - rejected, unnecessary complexity |
||||
- Cache key: package name, value: PackageInfo struct |
||||
|
||||
3. **Batch MarkExplicit**: Accept `[]string` packages, pass all to single `pacman -D --explicit` call |
||||
- Note: pacman -D accepts multiple packages in single call |
||||
|
||||
4. **Batch query strategy**: |
||||
- Query all packages against dyalpm local DB → returns found[] |
||||
- Query not-found against dyalpm sync DBs → returns found[] |
||||
- Query remaining not-found against AUR HTTP (single batched request) |
||||
- Current: pacman -Qip → pacman -Sip → AUR (per-package) |
||||
- New: Batch dyalpm local → Batch dyalpm sync → Batch AUR |
||||
|
||||
5. **Fallback for dyalpm unavailable**: If dyalpm init fails, fall back to: |
||||
- pacman -Qip for all packages (single process, capture all) |
||||
- pacman -Sip for remaining (single process, capture all) |
||||
- AUR HTTP batch for remaining (already batched) |
||||
|
||||
## Risks / Trade-offs |
||||
|
||||
- **Risk**: dyalpm requires libalpm.so.15 on system |
||||
- Mitigation: Check at runtime, fallback to process spawn if missing |
||||
|
||||
- **Risk**: Cache invalidation edge cases (e.g., package installed during job) |
||||
- Mitigation: Acceptable for declpac use case; user runs sync after config changes |
||||
|
||||
- **Risk**: AUR API batch size limits |
||||
- Mitigation: Chunk large batches if AUR has limits (TBD in implementation) |
||||
|
||||
## Migration Plan |
||||
|
||||
1. Add dyalpm dependency to go.mod |
||||
2. Refactor Pac struct: replace aurCache with unified pkgCache map |
||||
3. Change ValidatePackage to ValidatePackages (slice input, batch processing) |
||||
4. Update MarkExplicit to accept slice |
||||
5. Batch AUR HTTP calls in ensureAURCache |
||||
6. Update call sites in Sync(), categorizePackages, DryRun() |
||||
7. Test with existing test suite |
||||
@ -0,0 +1,28 @@
|
||||
## Why |
||||
|
||||
Current `ValidatePackage` spawns multiple `pacman` processes per package (-Qip, -Sip calls), causing performance issues when checking many packages. Additionally, `MarkExplicit` is called individually for each package, spawning a separate process per package. Finally, AUR HTTP calls are made per-package. Using the Jguer/dyalpm Go library eliminates process spawning overhead and enables efficient batch operations throughout. |
||||
|
||||
## What Changes |
||||
|
||||
- Replace shell-out to `pacman -Qip`/`pacman -Sip` in `ValidatePackage` with Jguer/dyalpm library calls |
||||
- Add unified in-memory package cache to `Pac` struct (replaces existing aurCache), persisting for job duration |
||||
- Replace individual `pacman -D --explicit` calls with single batch call |
||||
- Batch AUR HTTP queries into single request for all packages not found in pacman databases |
||||
- Add Jguer/dyalpm dependency |
||||
|
||||
## Capabilities |
||||
|
||||
### New Capabilities |
||||
|
||||
- **dyalpm-package-query**: Use dyalpm library for querying pacman package databases in batch, falling back to AUR HTTP batch for remaining packages. Results cached for job duration. |
||||
- **batch-explicit-mark**: Batch multiple packages into single `pacman -D --explicit` call instead of per-package process spawn. |
||||
|
||||
### Modified Capabilities |
||||
|
||||
None. |
||||
|
||||
## Impact |
||||
|
||||
- **pkg/pacman/pacman.go**: Refactor `ValidatePackage` to accept slice, replace aurCache with unified pkgCache, refactor `MarkExplicit` to accept slice |
||||
- **go.mod**: Add Jguer/dyalpm import |
||||
- **Performance**: Reduced process spawns from O(n*2 + n AUR) to O(1) for ValidatePackages, O(n) to O(1) for MarkExplicit |
||||
@ -0,0 +1,16 @@
|
||||
## ADDED Requirements |
||||
|
||||
### Requirement: MarkExplicit accepts multiple packages |
||||
The MarkExplicit method SHALL accept a slice of package names and mark all of them as explicitly installed using a single pacman -D call. |
||||
|
||||
#### Scenario: Single package marked explicit |
||||
- **WHEN** MarkExplicit is called with one package name |
||||
- **THEN** pacman -D --explicit is called once with that package |
||||
|
||||
#### Scenario: Multiple packages marked explicit |
||||
- **WHEN** MarkExplicit is called with multiple package names (e.g., ["pkg1", "pkg2"]) |
||||
- **THEN** pacman -D --explicit is called once with all packages as arguments |
||||
|
||||
#### Scenario: Empty package slice |
||||
- **WHEN** MarkExplicit is called with an empty slice |
||||
- **THEN** no pacman call is made, method returns nil immediately |
||||
@ -0,0 +1,47 @@
|
||||
## ADDED Requirements |
||||
|
||||
### Requirement: ValidatePackages uses dyalpm library with batch AUR fallback |
||||
The ValidatePackages method SHALL accept a slice of package names, query them in batch using dyalpm, then batch query AUR for any not found in pacman databases. |
||||
|
||||
#### Scenario: All packages found in local database |
||||
- **WHEN** ValidatePackages is called with packages that all exist in local pacman database |
||||
- **THEN** returns a slice of PackageInfo with Exists=true, InAUR=false for each |
||||
|
||||
#### Scenario: Some packages found in sync database |
||||
- **WHEN** ValidatePackages is called with packages where some are only in sync databases |
||||
- **THEN** returns PackageInfo with Exists=true, InAUR=false for found packages |
||||
|
||||
#### Scenario: Some packages found in AUR |
||||
- **WHEN** ValidatePackages is called and some packages are not in pacman databases |
||||
- **THEN** those packages are batched to single AUR HTTP request, returns PackageInfo with InAUR=true |
||||
|
||||
#### Scenario: Packages not found anywhere |
||||
- **WHEN** ValidatePackages is called and some packages not found in pacman or AUR |
||||
- **THEN** returns PackageInfo with Exists=false, InAUR=false for those packages |
||||
|
||||
### Requirement: Unified package cache |
||||
The Pac struct SHALL maintain a single cache map that stores all package query results (both pacman and AUR), replacing the separate aurCache. |
||||
|
||||
#### Scenario: Package already cached |
||||
- **WHEN** ValidatePackages is called for a package already queried in current job |
||||
- **THEN** cached result returned without any dyalpm or AUR calls |
||||
|
||||
#### Scenario: New Pac instance |
||||
- **WHEN** New() creates a new Pac instance |
||||
- **THEN** cache is empty (fresh job) |
||||
|
||||
### Requirement: Single batch AUR call |
||||
The AUR HTTP API SHALL be called once with all package names not found in pacman databases. |
||||
|
||||
#### Scenario: Multiple packages not in pacman |
||||
- **WHEN** 3 packages not found in local or sync databases |
||||
- **THEN** single AUR HTTP request with all 3 in arg[] params |
||||
- **AND** results cached for all 3 |
||||
|
||||
### Requirement: Fallback to process spawn if dyalpm unavailable |
||||
If dyalpm library is not available at runtime, ValidatePackages SHALL fall back to spawning pacman -Qip/-Sip processes in batch. |
||||
|
||||
#### Scenario: dyalpm unavailable |
||||
- **WHEN** dyalpm initialization fails |
||||
- **THEN** uses pacman -Qip for local, -Sip for sync, AUR HTTP batch as fallback |
||||
- **AND** behavior is identical to using dyalpm |
||||
@ -0,0 +1,36 @@
|
||||
## 1. Dependencies |
||||
|
||||
- [ ] 1.1 Add github.com/Jguer/dyalpm to go.mod |
||||
- [ ] 1.2 Run go mod tidy to resolve dependencies |
||||
|
||||
## 2. Pac Struct Updates |
||||
|
||||
- [ ] 2.1 Add dyalpm handle field to Pac struct |
||||
- [ ] 2.2 Add package info cache field (map[string]PackageInfo) to Pac struct |
||||
- [ ] 2.3 Update New() to initialize dyalpm handle |
||||
- [ ] 2.4 Update Close() to release dyalpm handle |
||||
|
||||
## 3. ValidatePackage Implementation |
||||
|
||||
- [ ] 3.1 Implement ValidatePackage using dyalpm LocalDB().Pkg() |
||||
- [ ] 3.2 Query sync databases if not found locally |
||||
- [ ] 3.3 Add caching logic to store/query results |
||||
- [ ] 3.4 Keep AUR fallback for packages not in pacman repos |
||||
|
||||
## 4. MarkExplicit Updates |
||||
|
||||
- [ ] 4.1 Change MarkExplicit signature to accept []string (slice of packages) |
||||
- [ ] 4.2 Implement single pacman -D --explicit call with all packages |
||||
- [ ] 4.3 Handle empty slice case (return nil immediately) |
||||
|
||||
## 5. Call Site Updates |
||||
|
||||
- [ ] 5.1 Update Sync() to pass packages slice to MarkExplicit |
||||
- [ ] 5.2 Ensure categorizePackages works with new ValidatePackage |
||||
|
||||
## 6. Testing |
||||
|
||||
- [ ] 6.1 Run existing tests to verify no regressions |
||||
- [ ] 6.2 Test ValidatePackage with local, sync, and AUR packages |
||||
- [ ] 6.3 Test MarkExplicit with single and multiple packages |
||||
- [ ] 6.4 Verify cache behavior (second call returns cached result) |
||||
Loading…
Reference in new issue