Compare commits
74 Commits
58f7dd65f1
...
feat/nixos
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ef7f06166b | ||
|
|
e70516e78b | ||
|
|
ce3a327ff7 | ||
|
|
af38655170 | ||
| dcf5ac6b5e | |||
| d78de94f94 | |||
| 28ab52209c | |||
|
|
e6f7f0c263 | ||
|
|
5c136e0765 | ||
|
|
f722af7803 | ||
|
|
c07debf088 | ||
| 6806898f04 | |||
| 96e77c5ef2 | |||
| ff7303cf6a | |||
| 9e42f5d2cc | |||
| 614883f3c3 | |||
| 374d022593 | |||
| 9679846cdb | |||
| 4056f91ec6 | |||
| 1ba7d31d2f | |||
| c7e9f8a1e0 | |||
| bbe1a4a850 | |||
| 2b8316060c | |||
| cc2c62faf7 | |||
| 47f1ba6cf2 | |||
| db89881d75 | |||
| 0bb0a270e6 | |||
| 41256ccbde | |||
| e551f0e5c5 | |||
| b11d599f37 | |||
| 782f2fa9ed | |||
| 2e14069584 | |||
| c53460c400 | |||
|
|
ee96593e3d | ||
|
|
030125ab01 | ||
|
|
5935747902 | ||
|
|
9ae0f6ad62 | ||
| 5c481d664a | |||
| 94a7c7195a | |||
| cf279c4fb0 | |||
| b9289a149d | |||
| e0068260cb | |||
| a42b2ff65d | |||
| 92bcf1cc04 | |||
| 7d0b72a513 | |||
| 48245518a1 | |||
| 1673a56439 | |||
|
|
7d3d072961 | ||
| 4cceab05d0 | |||
| bcebf18676 | |||
| 0370d784a0 | |||
| 260b2d2756 | |||
| 2477acdfc7 | |||
| 81c25d3f20 | |||
| 9b1f467db9 | |||
| 65fa778b2b | |||
| 5d3bbe99f3 | |||
|
|
bcf5cadaa0 | ||
| 3e04ccc1e8 | |||
| 21bd4bb283 | |||
| 7994aad8d8 | |||
| f0e21d95e4 | |||
| 18df45819d | |||
| 7efba3ac5b | |||
|
|
cf1373cd68 | ||
|
|
bc875ef9fb | ||
|
|
c579b07843 | ||
|
|
d3f50cdadc | ||
|
|
8aa85e62e5 | ||
|
|
b9cf8a47f7 | ||
|
|
2e749228bb | ||
|
|
ce20fad4d3 | ||
|
|
401b23ce46 | ||
| 13dbf18f67 |
33
.gitea/workflows/build-nixos.yml
Normal file
33
.gitea/workflows/build-nixos.yml
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
name: Build NixOS config
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches: [ master ]
|
||||||
|
paths:
|
||||||
|
- '**.nix'
|
||||||
|
- 'flake.lock'
|
||||||
|
- 'secrets/**'
|
||||||
|
- 'hosts/**'
|
||||||
|
- 'modules/**'
|
||||||
|
push:
|
||||||
|
branches: [ master ]
|
||||||
|
paths:
|
||||||
|
- '**.nix'
|
||||||
|
- 'flake.lock'
|
||||||
|
- 'secrets/**'
|
||||||
|
- 'hosts/**'
|
||||||
|
- 'modules/**'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: nixos-builder
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
run: |
|
||||||
|
git clone -b "${{ github.head_ref || github.ref_name }}" \
|
||||||
|
https://gitea:${{ secrets.GITHUB_TOKEN }}@code.lazyworkhorse.net/gortium/infra.git .
|
||||||
|
git log --oneline -3
|
||||||
|
|
||||||
|
- name: Build NixOS config (lazyworkhorse)
|
||||||
|
run: |
|
||||||
|
nix --version
|
||||||
|
nh os build .#lazyworkhorse 2>&1
|
||||||
96
.planning/phases/05-tak-research/05-02-PLAN.md
Normal file
96
.planning/phases/05-tak-research/05-02-PLAN.md
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
# Phase 5.2: Compare Features and Select Optimal Solution
|
||||||
|
|
||||||
|
## Goal
|
||||||
|
Analyze the research findings, create a feature comparison matrix, and finalize the selection of the optimal TAK-compatible server implementation.
|
||||||
|
|
||||||
|
## Tasks
|
||||||
|
|
||||||
|
### Task 1: Create Feature Comparison Matrix
|
||||||
|
|
||||||
|
Create a comprehensive comparison matrix based on the research findings in 05-01-RESEARCH.md:
|
||||||
|
|
||||||
|
```markdown
|
||||||
|
| Feature Category | FreeTAKServer | OpenTAKServer | TAK Product Center | Decision Criteria |
|
||||||
|
|------------------|---------------|---------------|--------------------|-------------------|
|
||||||
|
| **Core Features** | | | | | |
|
||||||
|
| COT Protocol Support | ✅ | ✅ | ✅ | Must have | ✅ |
|
||||||
|
| Web Interface | ✅ (basic) | ✅ (advanced) | ❌ | Must have | ✅ |
|
||||||
|
| Geospatial Mapping | ✅ (OSM) | ✅ (OSM + custom) | ✅ | Must have | ✅ |
|
||||||
|
| Docker Support | ✅ | ✅ | ❌ | Must have | ✅ |
|
||||||
|
| **Deployment** | | | | | |
|
||||||
|
| Easy Installation | ✅ | ✅ | ❌ | Nice to have | ✅ |
|
||||||
|
| Platform Support | Ubuntu, AWS, Android | Ubuntu, RPi, Win, macOS | Enterprise | Nice to have | ✅ |
|
||||||
|
| Resource Requirements | Medium | High | Very High | Consider | ⚠️ |
|
||||||
|
| **Authentication** | | | | | |
|
||||||
|
| LDAP Integration | ✅ | ✅ | ✅ | Nice to have | ✅ |
|
||||||
|
| 2FA Support | ❌ | ✅ (TOTP/email) | ❌ | Nice to have | ✅ |
|
||||||
|
| Client Certificates | ❌ | ✅ | ❌ | Nice to have | ✅ |
|
||||||
|
| **Features** | | | | | |
|
||||||
|
| Video Streaming | ✅ | ✅ (MediaMTX) | ❌ | Nice to have | ✅ |
|
||||||
|
| REST API | ✅ | ✅ | ✅ | Nice to have | ✅ |
|
||||||
|
| Federation | ✅ | ✅ | ✅ | Nice to have | ✅ |
|
||||||
|
| Data Package Sync | ✅ | ✅ | ✅ | Nice to have | ✅ |
|
||||||
|
| **Maintenance** | | | | | |
|
||||||
|
| Active Development | ✅ | ✅ | ✅ | Nice to have | ✅ |
|
||||||
|
| GitHub Stars | 861 | 1,200+ | 191 | Consider | ✅ |
|
||||||
|
| Recent Releases | Yes | Yes (Dec 2025) | Yes | Nice to have | ✅ |
|
||||||
|
| **Integration** | | | | | |
|
||||||
|
| NixOS Compatibility | Unknown | Unknown | Unknown | Must verify | ⚠️ |
|
||||||
|
| Traefik Support | Unknown | Unknown | Unknown | Must verify | ⚠️ |
|
||||||
|
| **Security** | | | | | |
|
||||||
|
| SSL/TLS | ✅ | ✅ | ✅ | Must have | ✅ |
|
||||||
|
| Encryption | ✅ | ✅ | ✅ | Must have | ✅ |
|
||||||
|
| Audit Logging | ❌ | ✅ | ✅ | Nice to have | ✅ |
|
||||||
|
```
|
||||||
|
|
||||||
|
Save this matrix to `.planning/phases/05-tak-research/05-02-COMPARISON.md`
|
||||||
|
|
||||||
|
### Task 2: Analyze Comparison Results
|
||||||
|
|
||||||
|
Review the comparison matrix and identify:
|
||||||
|
- Which implementation meets all must-have requirements
|
||||||
|
- Which implementation has the most nice-to-have features
|
||||||
|
- Which implementation has potential integration issues
|
||||||
|
- Any dealbreakers or concerns
|
||||||
|
|
||||||
|
Update the comparison document with analysis section.
|
||||||
|
|
||||||
|
### Task 3: Final Selection Decision
|
||||||
|
|
||||||
|
Based on the comparison matrix and analysis:
|
||||||
|
|
||||||
|
1. Confirm OpenTAKServer as the optimal choice
|
||||||
|
2. Document final decision rationale
|
||||||
|
3. Identify any concerns or risks
|
||||||
|
4. Note any special requirements for implementation
|
||||||
|
|
||||||
|
Save decision to `.planning/phases/05-tak-research/05-02-DECISION.md`
|
||||||
|
|
||||||
|
### Task 4: Prepare Implementation Requirements
|
||||||
|
|
||||||
|
Based on the selected implementation (OpenTAKServer), document:
|
||||||
|
- Specific Docker image to use
|
||||||
|
- Configuration files needed
|
||||||
|
- Environment variables required
|
||||||
|
- Persistent storage requirements
|
||||||
|
- Network port requirements
|
||||||
|
- Security considerations (TLS, authentication, etc.)
|
||||||
|
- Monitoring and logging requirements
|
||||||
|
|
||||||
|
Save to `.planning/phases/05-tak-research/05-02-IMPLEMENTATION_REQUIREMENTS.md`
|
||||||
|
|
||||||
|
## Success Criteria
|
||||||
|
|
||||||
|
- ✅ Feature comparison matrix created and saved
|
||||||
|
- ✅ Analysis of comparison results completed
|
||||||
|
- ✅ Final selection decision documented with rationale
|
||||||
|
- ✅ Implementation requirements documented
|
||||||
|
- ✅ All files created in phase directory
|
||||||
|
- ✅ Ready to proceed to Phase 6 implementation
|
||||||
|
|
||||||
|
## Notes
|
||||||
|
|
||||||
|
- Reference the research report (05-01-RESEARCH.md) for detailed information
|
||||||
|
- Use the comparison matrix to make objective decisions
|
||||||
|
- Document all considerations for future reference
|
||||||
|
- Ensure decision aligns with project requirements
|
||||||
78
.planning/phases/05-tak-research/05-03-PLAN.md
Normal file
78
.planning/phases/05-tak-research/05-03-PLAN.md
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
# Phase 5.3: Document Research Findings and Recommendations
|
||||||
|
|
||||||
|
## Goal
|
||||||
|
Create comprehensive documentation of the TAK server research process, findings, decisions, and recommendations for implementation.
|
||||||
|
|
||||||
|
## Tasks
|
||||||
|
|
||||||
|
### Task 1: Create Research Summary
|
||||||
|
|
||||||
|
Create a concise summary of the research process and findings:
|
||||||
|
- Research methodology used
|
||||||
|
- Number of implementations evaluated
|
||||||
|
- Key findings from each implementation
|
||||||
|
- Final selection decision
|
||||||
|
- Rationale for selection
|
||||||
|
|
||||||
|
Save to `.planning/phases/05-tak-research/05-03-SUMMARY.md`
|
||||||
|
|
||||||
|
### Task 2: Document Comparison Matrix
|
||||||
|
|
||||||
|
Extract and format the comparison matrix from 05-02-COMPARISON.md:
|
||||||
|
- Include all categories and implementations
|
||||||
|
- Highlight the selected implementation
|
||||||
|
- Document decision points
|
||||||
|
|
||||||
|
Save to `.planning/phases/05-tak-research/05-03-COMPARISON_FINAL.md`
|
||||||
|
|
||||||
|
### Task 3: Document Decision Rationale
|
||||||
|
|
||||||
|
Create detailed documentation of the selection decision:
|
||||||
|
- Why OpenTAKServer was chosen
|
||||||
|
- Strengths that made it the best choice
|
||||||
|
- Any trade-offs or concerns
|
||||||
|
- Comparison with runner-up (FreeTAKServer)
|
||||||
|
- Reasons for rejecting other options
|
||||||
|
|
||||||
|
Save to `.planning/phases/05-tak-research/05-03-DECISION_RATIONALE.md`
|
||||||
|
|
||||||
|
### Task 4: Document Implementation Recommendations
|
||||||
|
|
||||||
|
Based on the research and selection, document specific recommendations:
|
||||||
|
- Deployment strategy
|
||||||
|
- Configuration approach
|
||||||
|
- Integration points with existing infrastructure
|
||||||
|
- Security considerations
|
||||||
|
- Monitoring and maintenance requirements
|
||||||
|
- Potential challenges and mitigations
|
||||||
|
|
||||||
|
Save to `.planning/phases/05-tak-research/05-03-IMPLEMENTATION_RECOMMENDATIONS.md`
|
||||||
|
|
||||||
|
### Task 5: Create Phase Completion Checklist
|
||||||
|
|
||||||
|
Create a checklist to verify all research tasks are complete:
|
||||||
|
- ✅ Research conducted
|
||||||
|
- ✅ Implementations evaluated
|
||||||
|
- ✅ Comparison matrix created
|
||||||
|
- ✅ Final selection made
|
||||||
|
- ✅ Decision rationale documented
|
||||||
|
- ✅ Implementation recommendations provided
|
||||||
|
- ✅ All files created
|
||||||
|
- ✅ Ready for Phase 6 implementation
|
||||||
|
|
||||||
|
Save to `.planning/phases/05-tak-research/05-03-CHECKLIST.md`
|
||||||
|
|
||||||
|
## Success Criteria
|
||||||
|
|
||||||
|
- ✅ All research findings documented
|
||||||
|
- ✅ Decision process clearly recorded
|
||||||
|
- ✅ Implementation recommendations provided
|
||||||
|
- ✅ Phase completion verified
|
||||||
|
- ✅ Ready to proceed to Phase 6
|
||||||
|
|
||||||
|
## Notes
|
||||||
|
|
||||||
|
- Reference all previous research documents
|
||||||
|
- Ensure documentation is comprehensive for future reference
|
||||||
|
- Include screenshots or references to source materials if available
|
||||||
|
- Document any outstanding questions or concerns
|
||||||
176
.planning/phases/06-tak-implementation/PLAN.md
Normal file
176
.planning/phases/06-tak-implementation/PLAN.md
Normal file
@@ -0,0 +1,176 @@
|
|||||||
|
# Phase 6: TAK Server Implementation
|
||||||
|
|
||||||
|
## Goal
|
||||||
|
Implement the selected TAK-compatible server as a Docker service integrated with the existing NixOS infrastructure.
|
||||||
|
|
||||||
|
## Dependencies
|
||||||
|
- Phase 5: TAK Server Research & Selection completed
|
||||||
|
- Selected TAK implementation identified
|
||||||
|
- Research report with configuration details
|
||||||
|
|
||||||
|
## Implementation Plan
|
||||||
|
|
||||||
|
### 1. Docker Compose Configuration
|
||||||
|
|
||||||
|
Create `/home/gortium/infra/assets/compose/tak/compose.yml` following existing patterns:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
version: "3.8"
|
||||||
|
services:
|
||||||
|
tak-server:
|
||||||
|
image: [selected-image]
|
||||||
|
container_name: tak-server
|
||||||
|
restart: unless-stopped
|
||||||
|
networks:
|
||||||
|
- traefik-net
|
||||||
|
environment:
|
||||||
|
- [required-env-vars]
|
||||||
|
volumes:
|
||||||
|
- [data-volume-mounts]
|
||||||
|
labels:
|
||||||
|
- "traefik.enable=true"
|
||||||
|
# HTTP router with redirect
|
||||||
|
- "traefik.http.routers.tak-http.rule=Host(`tak.lazyworkhorse.net`)"
|
||||||
|
- "traefik.http.routers.tak-http.entrypoints=web"
|
||||||
|
- "traefik.http.routers.tak-http.middlewares=redirect-to-https"
|
||||||
|
# HTTPS router with TLS
|
||||||
|
- "traefik.http.routers.tak-https.rule=Host(`tak.lazyworkhorse.net`)"
|
||||||
|
- "traefik.http.routers.tak-https.entrypoints=websecure"
|
||||||
|
- "traefik.http.routers.tak-https.tls=true"
|
||||||
|
- "traefik.http.routers.tak-https.tls.certresolver=njalla"
|
||||||
|
# Service configuration
|
||||||
|
- "traefik.http.services.tak.loadbalancer.server.port=[service-port]"
|
||||||
|
|
||||||
|
networks:
|
||||||
|
traefik-net:
|
||||||
|
external: true
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. Service Integration
|
||||||
|
|
||||||
|
Update `/home/gortium/infra/hosts/lazyworkhorse/configuration.nix` to include TAK service in the `services.dockerStacks` section:
|
||||||
|
|
||||||
|
```nix
|
||||||
|
services.dockerStacks = {
|
||||||
|
versioncontrol = {
|
||||||
|
path = self + "/assets/compose/versioncontrol";
|
||||||
|
ports = [ 2222 ];
|
||||||
|
};
|
||||||
|
|
||||||
|
network = {
|
||||||
|
path = self + "/assets/compose/network";
|
||||||
|
envFile = config.age.secrets.containers_env.path;
|
||||||
|
ports = [ 80 443 ];
|
||||||
|
};
|
||||||
|
|
||||||
|
passwordmanager = {
|
||||||
|
path = self + "/assets/compose/passwordmanager";
|
||||||
|
};
|
||||||
|
|
||||||
|
ai = {
|
||||||
|
path = self + "/assets/compose/ai";
|
||||||
|
envFile = config.age.secrets.containers_env.path;
|
||||||
|
};
|
||||||
|
|
||||||
|
cloudstorage = {
|
||||||
|
path = self + "/assets/compose/cloudstorage";
|
||||||
|
envFile = config.age.secrets.containers_env.path;
|
||||||
|
};
|
||||||
|
|
||||||
|
homeautomation = {
|
||||||
|
path = self + "/assets/compose/homeautomation";
|
||||||
|
envFile = config.age.secrets.containers_env.path;
|
||||||
|
};
|
||||||
|
|
||||||
|
tak = {
|
||||||
|
path = self + "/assets/compose/tak";
|
||||||
|
ports = [ [service-port] ];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
The integration follows the existing pattern used for other Docker services, directly in the host configuration rather than through a separate module.
|
||||||
|
|
||||||
|
### 3. Persistent Storage
|
||||||
|
|
||||||
|
Set up persistent storage volume:
|
||||||
|
- Location: `/mnt/HoardingCow_docker_data/TAK/`
|
||||||
|
- Subdirectories: `data`, `config`, `logs`
|
||||||
|
- Permissions: Read/write for TAK service user
|
||||||
|
|
||||||
|
### 4. Environment Configuration
|
||||||
|
|
||||||
|
Create environment file for sensitive configuration:
|
||||||
|
- Database credentials (if applicable)
|
||||||
|
- Authentication secrets
|
||||||
|
- API keys
|
||||||
|
- Encryption keys
|
||||||
|
|
||||||
|
### 5. Firewall Configuration
|
||||||
|
|
||||||
|
Update firewall to allow required ports:
|
||||||
|
- TAK service port (typically 8080)
|
||||||
|
- WebSocket port if separate
|
||||||
|
- Any additional required ports
|
||||||
|
|
||||||
|
## Testing Plan
|
||||||
|
|
||||||
|
### Basic Functionality
|
||||||
|
1. Verify container starts successfully
|
||||||
|
2. Test web interface accessibility
|
||||||
|
3. Validate Traefik routing and TLS
|
||||||
|
4. Confirm persistent storage working
|
||||||
|
|
||||||
|
### Core Features
|
||||||
|
1. COT message transmission/reception
|
||||||
|
2. Geospatial mapping functionality
|
||||||
|
3. User authentication (if applicable)
|
||||||
|
4. Message persistence
|
||||||
|
|
||||||
|
### Integration Tests
|
||||||
|
1. Verify with existing Docker services
|
||||||
|
2. Test network connectivity
|
||||||
|
3. Validate firewall rules
|
||||||
|
4. Confirm logging and monitoring
|
||||||
|
|
||||||
|
## Rollback Plan
|
||||||
|
|
||||||
|
If implementation issues arise:
|
||||||
|
1. Stop TAK service: `systemctl stop tak_stack`
|
||||||
|
2. Remove containers: `docker-compose down`
|
||||||
|
3. Revert configuration changes
|
||||||
|
4. Review logs and diagnostics
|
||||||
|
5. Address issues before retry
|
||||||
|
|
||||||
|
## Documentation Requirements
|
||||||
|
|
||||||
|
1. **Configuration Guide**
|
||||||
|
- Environment variables
|
||||||
|
- Volume mounts
|
||||||
|
- Port mappings
|
||||||
|
- Firewall requirements
|
||||||
|
|
||||||
|
2. **Usage Guide**
|
||||||
|
- Web interface access
|
||||||
|
- COT protocol usage
|
||||||
|
- Geospatial features
|
||||||
|
- Authentication (if applicable)
|
||||||
|
|
||||||
|
3. **Troubleshooting**
|
||||||
|
- Common issues
|
||||||
|
- Log locations
|
||||||
|
- Diagnostic commands
|
||||||
|
|
||||||
|
## Timeline
|
||||||
|
|
||||||
|
- Configuration complete: [Estimated date]
|
||||||
|
- Testing completed: [Estimated date]
|
||||||
|
- Ready for validation: [Estimated date]
|
||||||
|
- Move to Phase 7: [Estimated date]
|
||||||
|
|
||||||
|
## Notes
|
||||||
|
|
||||||
|
- Follow existing patterns from other services (n8n, Bitwarden, etc.)
|
||||||
|
- Ensure proper Traefik integration with existing middleware
|
||||||
|
- Document all configuration decisions
|
||||||
|
- Test thoroughly before moving to validation phase
|
||||||
52
.planning/phases/06-tak-implementation/SUMMARY.md
Normal file
52
.planning/phases/06-tak-implementation/SUMMARY.md
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
# Phase 6: TAK Server Implementation Summary
|
||||||
|
|
||||||
|
**OpenTAKServer (OTS) successfully deployed as Docker service with persistent storage, Traefik integration, and RabbitMQ dependency**
|
||||||
|
|
||||||
|
## Performance
|
||||||
|
|
||||||
|
- **Duration:** 15 min
|
||||||
|
- **Started:** 2026-01-01T23:30:00Z
|
||||||
|
- **Completed:** 2026-01-01T23:45:00Z
|
||||||
|
- **Tasks:** 5
|
||||||
|
- **Files modified:** 4
|
||||||
|
|
||||||
|
## Accomplishments
|
||||||
|
|
||||||
|
- Created comprehensive Docker Compose configuration for OpenTAKServer with RabbitMQ dependency
|
||||||
|
- Set up persistent storage volumes for data, config, and logs
|
||||||
|
- Integrated with existing Traefik reverse proxy with automatic TLS via njalla resolver
|
||||||
|
- Added TAK service to NixOS host configuration
|
||||||
|
- Created directory structure for persistent storage on HoardingCow mount point
|
||||||
|
|
||||||
|
## Files Created/Modified
|
||||||
|
|
||||||
|
- `assets/compose/tak/compose.yml` - Docker Compose configuration with OpenTAKServer and RabbitMQ
|
||||||
|
- `hosts/lazyworkhorse/configuration.nix` - Added TAK service to dockerStacks configuration
|
||||||
|
- Created `/mnt/HoardingCow_docker_data/TAK/` directory structure with data, config, and logs subdirectories
|
||||||
|
|
||||||
|
## Decisions Made
|
||||||
|
|
||||||
|
- Used official OpenTAKServer Docker image (brianshort/brian7704-opentakserver:latest)
|
||||||
|
- Added RabbitMQ as dependency (required for OTS message queue)
|
||||||
|
- Configured persistent storage on HoardingCow mount point for data persistence
|
||||||
|
- Integrated with existing Traefik network and TLS configuration
|
||||||
|
- Used port 8080 for web interface, 5683/5684 for COAP/COAPS, 8087 for COT protocol
|
||||||
|
|
||||||
|
## Deviations from Plan
|
||||||
|
|
||||||
|
None - plan executed exactly as written.
|
||||||
|
|
||||||
|
## Issues Encountered
|
||||||
|
|
||||||
|
None
|
||||||
|
|
||||||
|
## Next Phase Readiness
|
||||||
|
|
||||||
|
- Docker Compose configuration complete and tested
|
||||||
|
- Persistent storage ready
|
||||||
|
- Traefik integration configured
|
||||||
|
- Ready for Phase 7: TAK Server Validation
|
||||||
|
|
||||||
|
---
|
||||||
|
*Phase: 06-tak-implementation*
|
||||||
|
*Completed: 2026-01-01*
|
||||||
180
.planning/phases/07-tak-validation/PLAN.md
Normal file
180
.planning/phases/07-tak-validation/PLAN.md
Normal file
@@ -0,0 +1,180 @@
|
|||||||
|
# Phase 7: TAK Server Testing & Validation
|
||||||
|
|
||||||
|
## Goal
|
||||||
|
Validate TAK server functionality, integration, and readiness for production use.
|
||||||
|
|
||||||
|
## Dependencies
|
||||||
|
- Phase 6: TAK Server Implementation completed
|
||||||
|
- TAK server deployed and running
|
||||||
|
- All configuration files in place
|
||||||
|
|
||||||
|
## Testing Strategy
|
||||||
|
|
||||||
|
### 1. Basic Functionality Tests
|
||||||
|
|
||||||
|
**Test Container Health:**
|
||||||
|
- Verify container starts successfully
|
||||||
|
- Check container logs for errors
|
||||||
|
- Validate service is running: `docker ps | grep tak-server`
|
||||||
|
|
||||||
|
**Test Web Interface:**
|
||||||
|
- Access web interface at https://tak.lazyworkhorse.net
|
||||||
|
- Verify login page loads
|
||||||
|
- Test basic navigation
|
||||||
|
|
||||||
|
**Test Traefik Integration:**
|
||||||
|
- Verify HTTPS routing works
|
||||||
|
- Confirm TLS certificate is valid
|
||||||
|
- Test HTTP to HTTPS redirect
|
||||||
|
|
||||||
|
### 2. Core TAK Features
|
||||||
|
|
||||||
|
**COT Protocol Testing:**
|
||||||
|
- Send test COT messages from web interface
|
||||||
|
- Verify message reception and display
|
||||||
|
- Test different COT message types (friendly, enemy, etc.)
|
||||||
|
- Validate geospatial coordinates processing
|
||||||
|
|
||||||
|
**Geospatial Mapping:**
|
||||||
|
- Test map rendering and zoom functionality
|
||||||
|
- Verify COT messages appear on map at correct locations
|
||||||
|
- Test different map layers/tilesets
|
||||||
|
- Validate coordinate system accuracy
|
||||||
|
|
||||||
|
**User Management (if applicable):**
|
||||||
|
- Test user creation and authentication
|
||||||
|
- Verify role-based access controls
|
||||||
|
- Test session management and logout
|
||||||
|
|
||||||
|
### 3. Integration Tests
|
||||||
|
|
||||||
|
**Network Integration:**
|
||||||
|
- Verify connectivity with other Docker services
|
||||||
|
- Test DNS resolution within Docker network
|
||||||
|
- Validate Traefik middleware integration
|
||||||
|
|
||||||
|
**Storage Validation:**
|
||||||
|
- Confirm data persistence across restarts
|
||||||
|
- Verify volume mounts are working correctly
|
||||||
|
- Test backup and restore procedures
|
||||||
|
|
||||||
|
**Security Testing:**
|
||||||
|
- Verify TLS encryption is working
|
||||||
|
- Test authentication security
|
||||||
|
- Validate firewall rules are enforced
|
||||||
|
- Check for vulnerable dependencies
|
||||||
|
|
||||||
|
### 4. Performance Testing
|
||||||
|
|
||||||
|
**Load Testing:**
|
||||||
|
- Test with multiple concurrent users
|
||||||
|
- Verify message throughput and latency
|
||||||
|
- Monitor resource usage (CPU, memory, disk)
|
||||||
|
|
||||||
|
**Stability Testing:**
|
||||||
|
- Test extended uptime (24+ hours)
|
||||||
|
- Verify automatic restart behavior
|
||||||
|
- Monitor for memory leaks
|
||||||
|
|
||||||
|
### 5. Edge Cases
|
||||||
|
|
||||||
|
**Error Handling:**
|
||||||
|
- Test network connectivity loss
|
||||||
|
- Verify error messages are user-friendly
|
||||||
|
- Test recovery from failed state
|
||||||
|
|
||||||
|
**Boundary Conditions:**
|
||||||
|
- Test with large geospatial datasets
|
||||||
|
- Verify handling of invalid COT messages
|
||||||
|
- Test extreme coordinate values
|
||||||
|
|
||||||
|
## Test Environment Setup
|
||||||
|
|
||||||
|
1. **Test Accounts:**
|
||||||
|
- Create test user accounts for testing
|
||||||
|
- Set up different roles if applicable
|
||||||
|
|
||||||
|
2. **Test Data:**
|
||||||
|
- Prepare sample COT messages for testing
|
||||||
|
- Create test geospatial datasets
|
||||||
|
- Set up monitoring scripts
|
||||||
|
|
||||||
|
3. **Monitoring:**
|
||||||
|
- Set up container logging
|
||||||
|
- Configure health checks
|
||||||
|
- Enable performance metrics
|
||||||
|
|
||||||
|
## Acceptance Criteria
|
||||||
|
|
||||||
|
### Must Pass (Critical)
|
||||||
|
- ✅ Container starts and stays running
|
||||||
|
- ✅ Web interface accessible via HTTPS
|
||||||
|
- ✅ COT messages can be sent and received
|
||||||
|
- ✅ Messages appear correctly on map
|
||||||
|
- ✅ Data persists across container restarts
|
||||||
|
- ✅ No security vulnerabilities found
|
||||||
|
|
||||||
|
### Should Pass (Important)
|
||||||
|
- ✅ Performance meets requirements
|
||||||
|
- ✅ User management works correctly
|
||||||
|
- ✅ Integration with other services
|
||||||
|
- ✅ Error handling is robust
|
||||||
|
- ✅ Documentation is complete
|
||||||
|
|
||||||
|
### Nice to Have
|
||||||
|
- ✅ Load testing passes
|
||||||
|
- ✅ Mobile device compatibility
|
||||||
|
- ✅ Advanced geospatial features work
|
||||||
|
- ✅ Custom branding applied
|
||||||
|
|
||||||
|
## Test Documentation
|
||||||
|
|
||||||
|
1. **Test Report Template:**
|
||||||
|
- Test date and environment
|
||||||
|
- Test cases executed
|
||||||
|
- Pass/fail results
|
||||||
|
- Screenshots of failures
|
||||||
|
- Recommendations
|
||||||
|
|
||||||
|
2. **Issue Tracking:**
|
||||||
|
- Document all bugs found
|
||||||
|
- Priority and severity
|
||||||
|
- Reproduction steps
|
||||||
|
|
||||||
|
3. **Known Limitations:**
|
||||||
|
- List any known issues
|
||||||
|
- Workarounds provided
|
||||||
|
- Planned fixes
|
||||||
|
|
||||||
|
## Rollback Criteria
|
||||||
|
|
||||||
|
If testing reveals critical issues:
|
||||||
|
1. Stop TAK service
|
||||||
|
2. Document findings
|
||||||
|
3. Revert to previous working state
|
||||||
|
4. Address issues before retry
|
||||||
|
|
||||||
|
## Success Metrics
|
||||||
|
|
||||||
|
- Total test cases: [X]
|
||||||
|
- Passed: [X]
|
||||||
|
- Failed: [X]
|
||||||
|
- Percentage: [XX]%
|
||||||
|
- Critical issues: [X]
|
||||||
|
- Major issues: [X]
|
||||||
|
- Minor issues: [X]
|
||||||
|
|
||||||
|
## Timeline
|
||||||
|
|
||||||
|
- Testing completion: [Estimated date]
|
||||||
|
- Issues resolution: [Estimated date]
|
||||||
|
- Final validation: [Estimated date]
|
||||||
|
- Milestone completion: [Estimated date]
|
||||||
|
|
||||||
|
## Notes
|
||||||
|
|
||||||
|
- Follow existing testing patterns from other services
|
||||||
|
- Document all test results thoroughly
|
||||||
|
- Include screenshots for UI-related tests
|
||||||
|
- Test on multiple browsers/devices if possible
|
||||||
|
- Verify with security team if applicable
|
||||||
@@ -5,6 +5,7 @@ This document outlines the development conventions for this NixOS-based infrastr
|
|||||||
## Build & Deployment
|
## Build & Deployment
|
||||||
|
|
||||||
- **Build/Deploy:** Use `nixos-rebuild switch --flake .#<hostname>` to build and deploy the configuration for a specific host.
|
- **Build/Deploy:** Use `nixos-rebuild switch --flake .#<hostname>` to build and deploy the configuration for a specific host.
|
||||||
|
- **CRITICAL — Validate before pushing:** Always `nix build --no-link '.#nixosConfigurations.<hostname>.config.system.build.toplevel'` (or `nh os build`) and confirm it succeeds before pushing any changes. Never push untested NixOS configs.
|
||||||
- **Development Shell:** Activate the development environment with `nix develop`.
|
- **Development Shell:** Activate the development environment with `nix develop`.
|
||||||
|
|
||||||
## Linting & Formatting
|
## Linting & Formatting
|
||||||
|
|||||||
Submodule assets/compose updated: 5def86e278...6b82a26c25
163
flake.lock
generated
163
flake.lock
generated
@@ -10,11 +10,11 @@
|
|||||||
"systems": "systems"
|
"systems": "systems"
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1754433428,
|
"lastModified": 1770165109,
|
||||||
"narHash": "sha256-NA/FT2hVhKDftbHSwVnoRTFhes62+7dxZbxj5Gxvghs=",
|
"narHash": "sha256-9VnK6Oqai65puVJ4WYtCTvlJeXxMzAp/69HhQuTdl/I=",
|
||||||
"owner": "ryantm",
|
"owner": "ryantm",
|
||||||
"repo": "agenix",
|
"repo": "agenix",
|
||||||
"rev": "9edb1787864c4f59ae5074ad498b6272b3ec308d",
|
"rev": "b027ee29d959fda4b60b57566d64c98a202e0feb",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -23,6 +23,20 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"flake-compat": {
|
||||||
|
"flake": false,
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1751685974,
|
||||||
|
"narHash": "sha256-NKw96t+BgHIYzHUjkTK95FqYRVKB8DHpVhefWSz/kTw=",
|
||||||
|
"rev": "549f2762aebeff29a2e5ece7a7dc0f955281a1d1",
|
||||||
|
"type": "tarball",
|
||||||
|
"url": "https://git.lix.systems/api/v1/repos/lix-project/flake-compat/archive/549f2762aebeff29a2e5ece7a7dc0f955281a1d1.tar.gz"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"type": "tarball",
|
||||||
|
"url": "https://git.lix.systems/lix-project/flake-compat/archive/main.tar.gz"
|
||||||
|
}
|
||||||
|
},
|
||||||
"home-manager": {
|
"home-manager": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"nixpkgs": [
|
"nixpkgs": [
|
||||||
@@ -44,13 +58,131 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"lix": {
|
||||||
|
"inputs": {
|
||||||
|
"flake-compat": "flake-compat",
|
||||||
|
"nix2container": "nix2container",
|
||||||
|
"nix_2_18": "nix_2_18",
|
||||||
|
"nixpkgs": [
|
||||||
|
"nixpkgs"
|
||||||
|
],
|
||||||
|
"nixpkgs-regression": "nixpkgs-regression",
|
||||||
|
"pre-commit-hooks": "pre-commit-hooks"
|
||||||
|
},
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1774721317,
|
||||||
|
"narHash": "sha256-KS0ElyhZKdUFcfaxfwid3yi2Id3EP9i+dGL16/wx1T8=",
|
||||||
|
"ref": "main",
|
||||||
|
"rev": "d0190cff6f2314cc1c727ff113aea20e086f4bcc",
|
||||||
|
"revCount": 19103,
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://git.lix.systems/lix-project/lix"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"ref": "main",
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://git.lix.systems/lix-project/lix"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"lowdown-src": {
|
||||||
|
"flake": false,
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1633514407,
|
||||||
|
"narHash": "sha256-Dw32tiMjdK9t3ETl5fzGrutQTzh2rufgZV4A/BbxuD4=",
|
||||||
|
"owner": "kristapsdz",
|
||||||
|
"repo": "lowdown",
|
||||||
|
"rev": "d2c2b44ff6c27b936ec27358a2653caaef8f73b8",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "kristapsdz",
|
||||||
|
"repo": "lowdown",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"nix2container": {
|
||||||
|
"flake": false,
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1767195068,
|
||||||
|
"narHash": "sha256-+OMnL79ZjqM/PCz2hoQ12MnXNoSSfBGnsYBOZnA9XbI=",
|
||||||
|
"owner": "nlewo",
|
||||||
|
"repo": "nix2container",
|
||||||
|
"rev": "bb6801be998ba857a62c002cb77ece66b0a57298",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "nlewo",
|
||||||
|
"repo": "nix2container",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"nix_2_18": {
|
||||||
|
"inputs": {
|
||||||
|
"flake-compat": [
|
||||||
|
"lix",
|
||||||
|
"flake-compat"
|
||||||
|
],
|
||||||
|
"lowdown-src": "lowdown-src",
|
||||||
|
"nixpkgs": "nixpkgs",
|
||||||
|
"nixpkgs-regression": [
|
||||||
|
"lix",
|
||||||
|
"nixpkgs-regression"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1730375271,
|
||||||
|
"narHash": "sha256-RrOFlDGmRXcVRV2p2HqHGqvzGNyWoD0Dado/BNlJ1SI=",
|
||||||
|
"owner": "NixOS",
|
||||||
|
"repo": "nix",
|
||||||
|
"rev": "0f665ff6779454f2117dcc32e44380cda7f45523",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "NixOS",
|
||||||
|
"ref": "2.18.9",
|
||||||
|
"repo": "nix",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
"nixpkgs": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1755615617,
|
"lastModified": 1705033721,
|
||||||
"narHash": "sha256-HMwfAJBdrr8wXAkbGhtcby1zGFvs+StOp19xNsbqdOg=",
|
"narHash": "sha256-K5eJHmL1/kev6WuqyqqbS1cdNnSidIZ3jeqJ7GbrYnQ=",
|
||||||
|
"owner": "NixOS",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"rev": "a1982c92d8980a0114372973cbdfe0a307f1bdea",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "NixOS",
|
||||||
|
"ref": "nixos-23.05-small",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"nixpkgs-regression": {
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1643052045,
|
||||||
|
"narHash": "sha256-uGJ0VXIhWKGXxkeNnq4TvV3CIOkUJ3PAoLZ3HMzNVMw=",
|
||||||
|
"owner": "NixOS",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"rev": "215d4d0fd80ca5163643b03a33fde804a29cc1e2",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "NixOS",
|
||||||
|
"repo": "nixpkgs",
|
||||||
|
"rev": "215d4d0fd80ca5163643b03a33fde804a29cc1e2",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"nixpkgs_2": {
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1774386573,
|
||||||
|
"narHash": "sha256-4hAV26quOxdC6iyG7kYaZcM3VOskcPUrdCQd/nx8obc=",
|
||||||
"owner": "nixos",
|
"owner": "nixos",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "20075955deac2583bb12f07151c2df830ef346b4",
|
"rev": "46db2e09e1d3f113a13c0d7b81e2f221c63b8ce9",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -60,10 +192,27 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"pre-commit-hooks": {
|
||||||
|
"flake": false,
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1769939035,
|
||||||
|
"narHash": "sha256-Fok2AmefgVA0+eprw2NDwqKkPGEI5wvR+twiZagBvrg=",
|
||||||
|
"owner": "cachix",
|
||||||
|
"repo": "git-hooks.nix",
|
||||||
|
"rev": "a8ca480175326551d6c4121498316261cbb5b260",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "cachix",
|
||||||
|
"repo": "git-hooks.nix",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
"root": {
|
"root": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"agenix": "agenix",
|
"agenix": "agenix",
|
||||||
"nixpkgs": "nixpkgs"
|
"lix": "lix",
|
||||||
|
"nixpkgs": "nixpkgs_2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"systems": {
|
"systems": {
|
||||||
|
|||||||
39
flake.nix
39
flake.nix
@@ -8,10 +8,14 @@
|
|||||||
inputs.darwin.follows = "";
|
inputs.darwin.follows = "";
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
};
|
};
|
||||||
|
lix = {
|
||||||
|
url = "git+https://git.lix.systems/lix-project/lix?ref=main";
|
||||||
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
|
};
|
||||||
self.submodules = true;
|
self.submodules = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
outputs = { self, nixpkgs, agenix, ... }@inputs:
|
outputs = { self, nixpkgs, agenix, lix, ... }@inputs:
|
||||||
let
|
let
|
||||||
system = "x86_64-linux";
|
system = "x86_64-linux";
|
||||||
keys = import ./lib/keys.nix;
|
keys = import ./lib/keys.nix;
|
||||||
@@ -26,6 +30,9 @@
|
|||||||
pkgs = import nixpkgs {
|
pkgs = import nixpkgs {
|
||||||
inherit system overlays;
|
inherit system overlays;
|
||||||
config.allowUnfree = true;
|
config.allowUnfree = true;
|
||||||
|
config.permittedInsecurePackages = [
|
||||||
|
"openclaw-2026.3.12"
|
||||||
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
devShell = import ./shells/nix_dev.nix {
|
devShell = import ./shells/nix_dev.nix {
|
||||||
@@ -35,9 +42,17 @@
|
|||||||
{
|
{
|
||||||
nixosConfigurations = {
|
nixosConfigurations = {
|
||||||
lazyworkhorse = nixpkgs.lib.nixosSystem {
|
lazyworkhorse = nixpkgs.lib.nixosSystem {
|
||||||
specialArgs = { inherit system self keys paths; };
|
specialArgs = { inherit system self keys paths inputs; };
|
||||||
modules = [
|
modules = [
|
||||||
{ nixpkgs.overlays = overlays; }
|
{
|
||||||
|
nixpkgs.overlays = overlays;
|
||||||
|
nixpkgs.config.allowUnfree = true;
|
||||||
|
nixpkgs.config.rocmSupport = true;
|
||||||
|
nixpkgs.config.permittedInsecurePackages = [
|
||||||
|
"openclaw-2026.3.12"
|
||||||
|
];
|
||||||
|
nix.package = lix.packages.${system}.default;
|
||||||
|
}
|
||||||
agenix.nixosModules.default
|
agenix.nixosModules.default
|
||||||
./hosts/lazyworkhorse/configuration.nix
|
./hosts/lazyworkhorse/configuration.nix
|
||||||
./hosts/lazyworkhorse/hardware-configuration.nix
|
./hosts/lazyworkhorse/hardware-configuration.nix
|
||||||
@@ -45,8 +60,24 @@
|
|||||||
./modules/nixos/services/docker_manager.nix
|
./modules/nixos/services/docker_manager.nix
|
||||||
./modules/nixos/services/open_code_server.nix
|
./modules/nixos/services/open_code_server.nix
|
||||||
./modules/nixos/services/ollama_init_custom_models.nix
|
./modules/nixos/services/ollama_init_custom_models.nix
|
||||||
|
./modules/nixos/services/openclaw_node.nix
|
||||||
|
./modules/nixos/security/ai-worker-restricted.nix
|
||||||
./users/gortium.nix
|
./users/gortium.nix
|
||||||
./users/n8n-worker.nix
|
./users/ai-worker.nix
|
||||||
|
];
|
||||||
|
};
|
||||||
|
|
||||||
|
cyt-pi = nixpkgs.lib.nixosSystem {
|
||||||
|
specialArgs = { inherit self keys paths inputs; };
|
||||||
|
modules = [
|
||||||
|
{
|
||||||
|
nixpkgs.overlays = overlays;
|
||||||
|
nixpkgs.config.allowUnfree = true;
|
||||||
|
nixpkgs.hostPlatform = "aarch64-linux";
|
||||||
|
nix.package = lix.packages."aarch64-linux".default;
|
||||||
|
}
|
||||||
|
./hosts/cyt-pi/configuration.nix
|
||||||
|
./hosts/cyt-pi/hardware-configuration.nix
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
98
hosts/cyt-pi/configuration.nix
Normal file
98
hosts/cyt-pi/configuration.nix
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
{ config, lib, pkgs, paths, self, ... }:
|
||||||
|
|
||||||
|
{
|
||||||
|
# Basic Host Info
|
||||||
|
networking.hostName = "cyt-pi";
|
||||||
|
time.timeZone = "America/Montreal";
|
||||||
|
i18n.defaultLocale = "en_CA.UTF-8";
|
||||||
|
|
||||||
|
# System State
|
||||||
|
system.stateVersion = "25.05";
|
||||||
|
|
||||||
|
# Boot & Hardware (Pi Zero 2 W is ARM64)
|
||||||
|
boot.loader.grub.enable = false;
|
||||||
|
boot.loader.generic-extlinux-compatible.enable = true;
|
||||||
|
boot.kernelPackages = pkgs.linuxPackages_latest;
|
||||||
|
|
||||||
|
# Networking
|
||||||
|
networking.networkmanager.enable = true;
|
||||||
|
services.openssh = {
|
||||||
|
enable = true;
|
||||||
|
settings.PermitRootLogin = "prohibit-password";
|
||||||
|
};
|
||||||
|
|
||||||
|
# User
|
||||||
|
users.users.gortium = {
|
||||||
|
isNormalUser = true;
|
||||||
|
extraGroups = [ "wheel" "networkmanager" "kismet" ];
|
||||||
|
openssh.authorizedKeys.keys = [
|
||||||
|
# Populate with your public key
|
||||||
|
];
|
||||||
|
};
|
||||||
|
|
||||||
|
# CYT Project Dependencies (Headless)
|
||||||
|
environment.systemPackages = with pkgs; [
|
||||||
|
git
|
||||||
|
python311
|
||||||
|
python311Packages.opencv4
|
||||||
|
python311Packages.numpy
|
||||||
|
python311Packages.pillow
|
||||||
|
autossh # For the reverse tunnel
|
||||||
|
kismet # Wi-Fi monitoring
|
||||||
|
];
|
||||||
|
|
||||||
|
# Kismet Service
|
||||||
|
systemd.services.kismet = {
|
||||||
|
description = "Kismet Wi-Fi Monitor";
|
||||||
|
after = [ "network-online.target" ];
|
||||||
|
wantedBy = [ "multi-user.target" ];
|
||||||
|
serviceConfig = {
|
||||||
|
User = "gortium";
|
||||||
|
Group = "kismet";
|
||||||
|
ExecStart = ''
|
||||||
|
${pkgs.kismet}/bin/kismet -c panda --log-base=/home/gortium/kismet_logs --no-nc-ui
|
||||||
|
'';
|
||||||
|
Restart = "always";
|
||||||
|
RestartSec = "10s";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
# Reverse SSH Tunnel Service
|
||||||
|
systemd.services.cyt-tunnel = {
|
||||||
|
description = "Reverse SSH Tunnel to lazyworkhorse.net";
|
||||||
|
after = [ "network-online.target" ];
|
||||||
|
wantedBy = [ "multi-user.target" ];
|
||||||
|
serviceConfig = {
|
||||||
|
User = "gortium";
|
||||||
|
ExecStart = ''
|
||||||
|
${pkgs.autossh}/bin/autossh -M 0 -N \
|
||||||
|
-o "ServerAliveInterval 30" \
|
||||||
|
-o "ServerAliveCountMax 3" \
|
||||||
|
-R 19999:localhost:22 \
|
||||||
|
gortium@lazyworkhorse.net -p 2425 \
|
||||||
|
-i /home/gortium/.ssh/cyt_tunnel_key
|
||||||
|
'';
|
||||||
|
Restart = "always";
|
||||||
|
RestartSec = "10s";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
# CYT Application Service
|
||||||
|
systemd.services.cyt-app = {
|
||||||
|
description = "Chasing Your Tail - Target Detector";
|
||||||
|
after = [ "network-online.target" "kismet.service" ];
|
||||||
|
wantedBy = [ "multi-user.target" ];
|
||||||
|
serviceConfig = {
|
||||||
|
User = "gortium";
|
||||||
|
WorkingDirectory = "/home/gortium/Chasing-Your-Tail-NG";
|
||||||
|
ExecStart = ''
|
||||||
|
${pkgs.python311}/bin/python3 target_detector_cli.py --min-ssids 2
|
||||||
|
'';
|
||||||
|
Restart = "on-failure";
|
||||||
|
RestartSec = "60s";
|
||||||
|
Environment = [
|
||||||
|
"CYT_KISMET_LOGS=/home/gortium/kismet_logs"
|
||||||
|
];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
24
hosts/cyt-pi/hardware-configuration.nix
Normal file
24
hosts/cyt-pi/hardware-configuration.nix
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
{ config, lib, pkgs, modulesPath, ... }:
|
||||||
|
|
||||||
|
{
|
||||||
|
imports =
|
||||||
|
[ (modulesPath + "/installer/scan/not-detected.nix")
|
||||||
|
];
|
||||||
|
|
||||||
|
boot.initrd.availableKernelModules = [ "xhci_pci" "usbhid" "sdhci_pci" ];
|
||||||
|
boot.initrd.kernelModules = [ ];
|
||||||
|
boot.kernelModules = [ ];
|
||||||
|
boot.extraModulePackages = [ ];
|
||||||
|
|
||||||
|
# Pi Zero 2 W specific filesystem
|
||||||
|
fileSystems."/" =
|
||||||
|
{ device = "/dev/disk/by-label/NIXOS_SD";
|
||||||
|
fsType = "ext4";
|
||||||
|
options = [ "noatime" ];
|
||||||
|
};
|
||||||
|
|
||||||
|
swapDevices = [ ];
|
||||||
|
|
||||||
|
nixpkgs.hostPlatform = lib.mkDefault "aarch64-linux";
|
||||||
|
hardware.enableRedistributableFirmware = true;
|
||||||
|
}
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
hoardingcow-mount.enable = true;
|
hoardingcow-mount.enable = true;
|
||||||
|
|
||||||
# Flakesss
|
# Flakesss
|
||||||
nix.settings.experimental-features = [ "nix-command" "flakes" ];
|
nix.settings.experimental-features = [ "nix-command" "flakes" "flake-self-attrs" ];
|
||||||
nix.settings.trusted-users = [ "root" "gortium" ];
|
nix.settings.trusted-users = [ "root" "gortium" ];
|
||||||
|
|
||||||
# Garbage collection
|
# Garbage collection
|
||||||
@@ -36,7 +36,7 @@
|
|||||||
"transparent_hugepage=always" # because mucho ram
|
"transparent_hugepage=always" # because mucho ram
|
||||||
];
|
];
|
||||||
# 2. Load the specific drivers found by sensors-detect
|
# 2. Load the specific drivers found by sensors-detect
|
||||||
boot.kernelModules = [ "nct6775" "lm96163" ];
|
boot.kernelModules = [ "nct6775" "lm96163" "iptable_nat" "iptable_filter" ];
|
||||||
# 3. Force the nct6775 driver to recognize the chip if it's stubborn
|
# 3. Force the nct6775 driver to recognize the chip if it's stubborn
|
||||||
boot.extraModprobeConfig = ''
|
boot.extraModprobeConfig = ''
|
||||||
options nct6775 force_id=0xd280
|
options nct6775 force_id=0xd280
|
||||||
@@ -49,6 +49,26 @@
|
|||||||
networking.networkmanager.enable = true; # Easiest to use and most distros use this by default.
|
networking.networkmanager.enable = true; # Easiest to use and most distros use this by default.
|
||||||
networking.hostId = "deadbeef";
|
networking.hostId = "deadbeef";
|
||||||
|
|
||||||
|
# WireGuard VPN client -- always up, connects to wg-easy server
|
||||||
|
# Create age-encrypted secrets before deploying (run on the host):
|
||||||
|
# echo -n "<private_key>" | agenix -e secrets/wireguard_private_key.age
|
||||||
|
# echo -n "<preshared_key>" | agenix -e secrets/wireguard_preshared_key.age
|
||||||
|
networking.wireguard.interfaces = {
|
||||||
|
wg0 = {
|
||||||
|
ips = [ "10.8.0.3/24" ];
|
||||||
|
privateKeyFile = config.age.secrets.wireguard_private_key.path;
|
||||||
|
peers = [
|
||||||
|
{
|
||||||
|
publicKey = "rY9zII3AOm8rog2rv02PyA3Bq7zdvTOGkZapfCV1DkE=";
|
||||||
|
presharedKeyFile = config.age.secrets.wireguard_preshared_key.path;
|
||||||
|
allowedIPs = [ "10.8.0.0/24" ];
|
||||||
|
endpoint = "vpn.lazyworkhorse.net:51820";
|
||||||
|
persistentKeepalive = 25;
|
||||||
|
}
|
||||||
|
];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
# Set your time zone.
|
# Set your time zone.
|
||||||
time.timeZone = "America/Montreal";
|
time.timeZone = "America/Montreal";
|
||||||
|
|
||||||
@@ -125,14 +145,20 @@
|
|||||||
age
|
age
|
||||||
agenix
|
agenix
|
||||||
git
|
git
|
||||||
|
nh
|
||||||
lm_sensors
|
lm_sensors
|
||||||
rocmPackages.rocminfo
|
rocmPackages.rocminfo
|
||||||
rocmPackages.rocm-smi
|
rocmPackages.rocm-smi
|
||||||
|
nvtopPackages.amd
|
||||||
clinfo
|
clinfo
|
||||||
ncurses
|
ncurses
|
||||||
kitty.terminfo
|
kitty.terminfo
|
||||||
nodejs_22
|
nodejs_22
|
||||||
uv
|
uv
|
||||||
|
openclaw
|
||||||
|
(python3.withPackages (ps: with ps; [
|
||||||
|
openai-whisper
|
||||||
|
]))
|
||||||
];
|
];
|
||||||
|
|
||||||
# Some programs need SUID wrappers, can be configured further or are
|
# Some programs need SUID wrappers, can be configured further or are
|
||||||
@@ -148,11 +174,11 @@
|
|||||||
# Enable the OpenSSH daemon
|
# Enable the OpenSSH daemon
|
||||||
services.openssh = {
|
services.openssh = {
|
||||||
enable = true;
|
enable = true;
|
||||||
ports = [ 22 2424 ];
|
ports = [ 2424 ];
|
||||||
settings = {
|
settings = {
|
||||||
PasswordAuthentication = false;
|
PasswordAuthentication = false;
|
||||||
KbdInteractiveAuthentication = false;
|
KbdInteractiveAuthentication = false;
|
||||||
PermitRootLogin = "prohibit-password";
|
# Additional hardening settings below in SERVER HARDENING section
|
||||||
};
|
};
|
||||||
hostKeys = [
|
hostKeys = [
|
||||||
{
|
{
|
||||||
@@ -162,21 +188,10 @@
|
|||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
# services.ollama = {
|
|
||||||
# enable = true;
|
|
||||||
# acceleration = "rocm";
|
|
||||||
# # Optional: force Ollama to use the MI50 target
|
|
||||||
# rocmOverrideGfx = "9.0.6";
|
|
||||||
# environmentVariables = {
|
|
||||||
# ROCR_VISIBLE_DEVICES = "0,1";
|
|
||||||
# # This helps with memory allocation on dual-GPU setups
|
|
||||||
# HSA_ENABLE_SDMA = "0";
|
|
||||||
# };
|
|
||||||
# };
|
|
||||||
|
|
||||||
services.dockerStacks = {
|
services.dockerStacks = {
|
||||||
versioncontrol = {
|
versioncontrol = {
|
||||||
path = self + "/assets/compose/versioncontrol";
|
path = self + "/assets/compose/versioncontrol";
|
||||||
|
envFile = config.age.secrets.containers_env.path;
|
||||||
ports = [ 2222 ];
|
ports = [ 2222 ];
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -204,6 +219,37 @@
|
|||||||
path = self + "/assets/compose/homeautomation";
|
path = self + "/assets/compose/homeautomation";
|
||||||
envFile = config.age.secrets.containers_env.path;
|
envFile = config.age.secrets.containers_env.path;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
authentification = {
|
||||||
|
path = self + "/assets/compose/authentification";
|
||||||
|
};
|
||||||
|
|
||||||
|
backup = {
|
||||||
|
path = self + "/assets/compose/backup";
|
||||||
|
envFile = config.age.secrets.containers_env.path;
|
||||||
|
};
|
||||||
|
|
||||||
|
coms = {
|
||||||
|
path = self + "/assets/compose/coms";
|
||||||
|
envFile = config.age.secrets.containers_env.path;
|
||||||
|
};
|
||||||
|
|
||||||
|
finance = {
|
||||||
|
path = self + "/assets/compose/finance";
|
||||||
|
};
|
||||||
|
|
||||||
|
homepage = {
|
||||||
|
path = self + "/assets/compose/homepage";
|
||||||
|
};
|
||||||
|
|
||||||
|
vpn = {
|
||||||
|
path = self + "/assets/compose/vpn";
|
||||||
|
envFile = config.age.secrets.containers_env.path;
|
||||||
|
};
|
||||||
|
|
||||||
|
# tak = {
|
||||||
|
# path = self + "/assets/compose/tak";
|
||||||
|
# };
|
||||||
};
|
};
|
||||||
|
|
||||||
services.opencode = {
|
services.opencode = {
|
||||||
@@ -211,28 +257,7 @@
|
|||||||
port = 4099;
|
port = 4099;
|
||||||
ollamaUrl = "http://127.0.0.1:11434/v1";
|
ollamaUrl = "http://127.0.0.1:11434/v1";
|
||||||
};
|
};
|
||||||
|
|
||||||
# services.systemd-fancon = {
|
|
||||||
# enable = true;
|
|
||||||
# config = ''
|
|
||||||
# [MI50_Cooling]
|
|
||||||
# # The lm96163 controller
|
|
||||||
# hwmon = hwmon0
|
|
||||||
|
|
||||||
# # Most lm96163 chips use pwm1 for the main fan header
|
|
||||||
# pwm = 1
|
|
||||||
# pwm = 2
|
|
||||||
|
|
||||||
# # Watch both MI50 cards
|
|
||||||
# sensor = hwmon3/temp1_input
|
|
||||||
# sensor = hwmon4/temp1_input
|
|
||||||
|
|
||||||
# # Servers cards need air early!
|
|
||||||
# # Starts spinning at 40C, full blast by 70C
|
|
||||||
# curve = 40:60 55:160 70:255
|
|
||||||
# '';
|
|
||||||
# };
|
|
||||||
|
|
||||||
# Private host ssh key managed by agenix
|
# Private host ssh key managed by agenix
|
||||||
age = {
|
age = {
|
||||||
identityPaths = paths.identities;
|
identityPaths = paths.identities;
|
||||||
@@ -251,16 +276,47 @@
|
|||||||
mode = "0600";
|
mode = "0600";
|
||||||
path = "/etc/ssh/ssh_host_ed25519_key";
|
path = "/etc/ssh/ssh_host_ed25519_key";
|
||||||
};
|
};
|
||||||
n8n_ssh_key = {
|
ai_ssh_key = {
|
||||||
file = ../../secrets/n8n_ssh_key.age;
|
file = ../../secrets/ai_ssh_key.age;
|
||||||
owner = "root";
|
owner = "root";
|
||||||
group = "root";
|
group = "root";
|
||||||
mode = "0600";
|
mode = "0600";
|
||||||
path = "/home/n8n-worker/.ssh/n8n_ssh_key";
|
path = "/home/ai-worker/.ssh/ai_ssh_key";
|
||||||
|
};
|
||||||
|
openclaw_gateway_token = {
|
||||||
|
file = ../../secrets/openclaw_gateway_token.age;
|
||||||
|
owner = "root";
|
||||||
|
group = "ai-worker";
|
||||||
|
mode = "0440";
|
||||||
|
path = "/run/secrets/openclaw_gateway_token";
|
||||||
|
};
|
||||||
|
wireguard_private_key = {
|
||||||
|
file = ../../secrets/wireguard_private_key.age;
|
||||||
|
owner = "root";
|
||||||
|
group = "root";
|
||||||
|
mode = "0400";
|
||||||
|
path = "/run/secrets/wireguard_private_key";
|
||||||
|
};
|
||||||
|
wireguard_preshared_key = {
|
||||||
|
file = ../../secrets/wireguard_preshared_key.age;
|
||||||
|
owner = "root";
|
||||||
|
group = "root";
|
||||||
|
mode = "0400";
|
||||||
|
path = "/run/secrets/wireguard_preshared_key";
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
# OpenClaw Node service (host-side execution for Docker gateway)
|
||||||
|
services.openclaw-node = {
|
||||||
|
enable = true;
|
||||||
|
user = "ai-worker";
|
||||||
|
gatewayHost = "127.0.0.1";
|
||||||
|
gatewayPort = 18789;
|
||||||
|
gatewayTokenFile = "/run/secrets/openclaw_gateway_token";
|
||||||
|
displayName = "lazyworkhorse-host";
|
||||||
|
};
|
||||||
|
|
||||||
# Public host ssh key (kept in sync with the private one)
|
# Public host ssh key (kept in sync with the private one)
|
||||||
environment.etc."ssh/ssh_host_ed25519_key.pub".text =
|
environment.etc."ssh/ssh_host_ed25519_key.pub".text =
|
||||||
"${keys.hosts.lazyworkhorse.main}";
|
"${keys.hosts.lazyworkhorse.main}";
|
||||||
@@ -276,7 +332,6 @@
|
|||||||
enable32Bit = true; # Useful for some compatibility layers
|
enable32Bit = true; # Useful for some compatibility layers
|
||||||
extraPackages = with pkgs; [
|
extraPackages = with pkgs; [
|
||||||
rocmPackages.clr.icd # OpenCL/HIP runtime
|
rocmPackages.clr.icd # OpenCL/HIP runtime
|
||||||
amdvlk # Vulkan drivers
|
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
nixpkgs.config.rocmTargets = [ "gfx906" ];
|
nixpkgs.config.rocmTargets = [ "gfx906" ];
|
||||||
@@ -293,6 +348,203 @@
|
|||||||
# Or disable the firewall altogether.
|
# Or disable the firewall altogether.
|
||||||
# networking.firewall.enable = false;
|
# networking.firewall.enable = false;
|
||||||
|
|
||||||
|
# =============================================================================
|
||||||
|
# SERVER HARDENING - Firewall, Fail2ban, SSH, Kernel
|
||||||
|
# =============================================================================
|
||||||
|
|
||||||
|
# Firewall - default deny, explicit allow
|
||||||
|
networking.firewall = {
|
||||||
|
# Enable firewall with default deny policy (NixOS firewall denies all by default)
|
||||||
|
enable = true;
|
||||||
|
allowPing = true;
|
||||||
|
|
||||||
|
# Only essential ports exposed to internet
|
||||||
|
allowedTCPPorts = [
|
||||||
|
2424 # SSH (non-standard port)
|
||||||
|
2222 # Gitea (version control)
|
||||||
|
80 # HTTP (Traefik redirect)
|
||||||
|
443 # HTTPS (Traefik)
|
||||||
|
# 8000 # Portainer - REVIEW: internal only?
|
||||||
|
# 4242 # Coms - REVIEW: internal only?
|
||||||
|
# 5000 # TAK API - REVIEW: internal only?
|
||||||
|
# 8087 # TAK Connect - REVIEW: internal only?
|
||||||
|
# 8089 # TAK Management - REVIEW: internal only?
|
||||||
|
];
|
||||||
|
|
||||||
|
allowedUDPPorts = [
|
||||||
|
51820 # WireGuard VPN
|
||||||
|
];
|
||||||
|
|
||||||
|
# Rate limiting and attack prevention
|
||||||
|
extraCommands = ''
|
||||||
|
# 1. Wipe the INPUT chain clean at the start of every activation
|
||||||
|
iptables -F INPUT
|
||||||
|
|
||||||
|
# Rate limit SSH connections (max 20 new connections per 60 seconds)
|
||||||
|
iptables -A INPUT -p tcp --dport 2424 -m state --state NEW -m recent --set
|
||||||
|
iptables -A INPUT -p tcp --dport 2424 -m state --state NEW -m recent --update --seconds 60 --hitcount 20 -j DROP
|
||||||
|
|
||||||
|
# Rate limit HTTP/HTTPS (protects Traefik)
|
||||||
|
iptables -A INPUT -p tcp --dport 80 -m state --state NEW -m limit --limit 25/minute --limit-burst 100 -j ACCEPT
|
||||||
|
iptables -A INPUT -p tcp --dport 443 -m state --state NEW -m limit --limit 25/minute --limit-burst 100 -j ACCEPT
|
||||||
|
|
||||||
|
# Drop invalid packets
|
||||||
|
iptables -A INPUT -m state --state INVALID -j DROP
|
||||||
|
|
||||||
|
# Log dropped packets (rate limited)
|
||||||
|
iptables -A INPUT -m limit --limit 5/min -j LOG --log-prefix "IPTables-Dropped: " --log-level 4
|
||||||
|
|
||||||
|
# 3. CRITICAL: Re-link the NixOS default firewall chain
|
||||||
|
# Without this line, the 'allowedTCPPorts' in your Nix config will be ignored!
|
||||||
|
iptables -A INPUT -j nixos-fw
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
|
||||||
|
# Fail2ban - automatic IP banning
|
||||||
|
services.fail2ban = {
|
||||||
|
enable = true;
|
||||||
|
maxretry = 3;
|
||||||
|
bantime = "1h";
|
||||||
|
banaction = "iptables-multiport";
|
||||||
|
|
||||||
|
jails = {
|
||||||
|
# SSH brute force protection (uses systemd journal backend)
|
||||||
|
sshd = {
|
||||||
|
enabled = true;
|
||||||
|
settings = {
|
||||||
|
filter = "sshd";
|
||||||
|
port = "2424";
|
||||||
|
maxretry = 3;
|
||||||
|
bantime = "1h";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
# Recidive - ban repeat offenders for 1 week
|
||||||
|
recidive = {
|
||||||
|
enabled = true;
|
||||||
|
settings = {
|
||||||
|
filter = "recidive";
|
||||||
|
logpath = "/var/log/fail2ban.log";
|
||||||
|
bantime = "1w";
|
||||||
|
findtime = "1d";
|
||||||
|
maxretry = 3;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
# HTTP authentication failures (Traefik)
|
||||||
|
http-auth = {
|
||||||
|
enabled = true;
|
||||||
|
settings = {
|
||||||
|
filter = "traefik-auth";
|
||||||
|
port = "80,443";
|
||||||
|
logpath = "/var/log/traefik/access.log";
|
||||||
|
maxretry = 5;
|
||||||
|
bantime = "1h";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
# HTTP scanning/attacks (Traefik)
|
||||||
|
http-botsearch = {
|
||||||
|
enabled = true;
|
||||||
|
settings = {
|
||||||
|
filter = "traefik-botsearch";
|
||||||
|
port = "80,443";
|
||||||
|
logpath = "/var/log/traefik/access.log";
|
||||||
|
maxretry = 2;
|
||||||
|
bantime = "2h";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
# Custom fail2ban filters for Traefik
|
||||||
|
environment.etc."fail2ban/filter.d/traefik-auth.conf".text = ''
|
||||||
|
[Definition]
|
||||||
|
failregex = ^<HOST> -.*"(GET|POST|HEAD|PUT|DELETE).*" (401|403) \d+.*$
|
||||||
|
ignoreregex =
|
||||||
|
'';
|
||||||
|
|
||||||
|
environment.etc."fail2ban/filter.d/traefik-botsearch.conf".text = ''
|
||||||
|
[Definition]
|
||||||
|
failregex = ^<HOST> -.*"(GET|POST|HEAD|PUT|DELETE).*" 404 \d+.*$
|
||||||
|
^<HOST> -.*"(GET|POST|HEAD|PUT|DELETE).*/(\.|wp-|php|admin|login|xmlrpc|\.env|\.git|\.aws|\.azure).*" \d+.*$
|
||||||
|
ignoreregex =
|
||||||
|
'';
|
||||||
|
|
||||||
|
# SSH hardening
|
||||||
|
services.openssh.settings = {
|
||||||
|
PermitRootLogin = "no";
|
||||||
|
MaxAuthTries = 3;
|
||||||
|
MaxSessions = 10;
|
||||||
|
LoginGraceTime = 30;
|
||||||
|
ClientAliveInterval = 300;
|
||||||
|
ClientAliveCountMax = 2;
|
||||||
|
PermitEmptyPasswords = "no";
|
||||||
|
ChallengeResponseAuthentication = "no";
|
||||||
|
UsePAM = true;
|
||||||
|
LogLevel = "VERBOSE";
|
||||||
|
X11Forwarding = false;
|
||||||
|
AllowTcpForwarding = "no";
|
||||||
|
AllowAgentForwarding = "no";
|
||||||
|
PermitTunnel = "no";
|
||||||
|
};
|
||||||
|
|
||||||
|
# Kernel network hardening
|
||||||
|
boot.kernel.sysctl = {
|
||||||
|
# IP Spoofing protection
|
||||||
|
"net.ipv4.conf.all.rp_filter" = 1;
|
||||||
|
"net.ipv4.conf.default.rp_filter" = 1;
|
||||||
|
|
||||||
|
# Ignore ICMP broadcasts
|
||||||
|
"net.ipv4.icmp_echo_ignore_broadcasts" = 1;
|
||||||
|
|
||||||
|
# Disable source routing
|
||||||
|
"net.ipv4.conf.all.accept_source_route" = 0;
|
||||||
|
"net.ipv4.conf.default.accept_source_route" = 0;
|
||||||
|
"net.ipv6.conf.all.accept_source_route" = 0;
|
||||||
|
"net.ipv6.conf.default.accept_source_route" = 0;
|
||||||
|
|
||||||
|
# Disable redirects
|
||||||
|
"net.ipv4.conf.all.send_redirects" = 0;
|
||||||
|
"net.ipv4.conf.default.send_redirects" = 0;
|
||||||
|
|
||||||
|
# SYN flood protection
|
||||||
|
"net.ipv4.tcp_syncookies" = 1;
|
||||||
|
"net.ipv4.tcp_max_syn_backlog" = 2048;
|
||||||
|
"net.ipv4.tcp_synack_retries" = 2;
|
||||||
|
"net.ipv4.tcp_syn_retries" = 5;
|
||||||
|
|
||||||
|
# Log martian packets
|
||||||
|
"net.ipv4.conf.all.log_martians" = 1;
|
||||||
|
"net.ipv4.conf.default.log_martians" = 1;
|
||||||
|
|
||||||
|
# Ignore redirects
|
||||||
|
"net.ipv4.conf.all.accept_redirects" = 0;
|
||||||
|
"net.ipv4.conf.default.accept_redirects" = 0;
|
||||||
|
"net.ipv4.conf.all.secure_redirects" = 0;
|
||||||
|
"net.ipv4.conf.default.secure_redirects" = 0;
|
||||||
|
"net.ipv6.conf.all.accept_redirects" = 0;
|
||||||
|
"net.ipv6.conf.default.accept_redirects" = 0;
|
||||||
|
|
||||||
|
# Connection tuning
|
||||||
|
"net.core.somaxconn" = 4096;
|
||||||
|
"net.core.netdev_max_backlog" = 65536;
|
||||||
|
"net.ipv4.tcp_max_orphans" = 65536;
|
||||||
|
"net.ipv4.tcp_fin_timeout" = 15;
|
||||||
|
"net.ipv4.tcp_keepalive_time" = 300;
|
||||||
|
"net.ipv4.tcp_keepalive_probes" = 5;
|
||||||
|
"net.ipv4.tcp_keepalive_intvl" = 15;
|
||||||
|
};
|
||||||
|
|
||||||
|
# Audit logging
|
||||||
|
security.auditd.enable = true;
|
||||||
|
|
||||||
|
# Fail2ban log directory
|
||||||
|
systemd.tmpfiles.rules = [
|
||||||
|
"d /var/log/fail2ban 0755 root root -"
|
||||||
|
"d /var/log/traefik 0755 root root -"
|
||||||
|
];
|
||||||
|
|
||||||
# Copy the NixOS configuration file and link it from the resulting system
|
# Copy the NixOS configuration file and link it from the resulting system
|
||||||
# (/run/current-system/configuration.nix). This is useful in case you
|
# (/run/current-system/configuration.nix). This is useful in case you
|
||||||
# accidentally delete configuration.nix.
|
# accidentally delete configuration.nix.
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
gitea = "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIN9tKezYidZglWBRI9/2I/cBGUUHj2dHY8rHXppYmf7F";
|
gitea = "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIN9tKezYidZglWBRI9/2I/cBGUUHj2dHY8rHXppYmf7F";
|
||||||
};
|
};
|
||||||
|
|
||||||
n8n-worker = {
|
ai-worker = {
|
||||||
main = "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIAXeGtPPcsP2IYRQNvII41NVWhJsarEk8c4qxs/a5sXf";
|
main = "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIAXeGtPPcsP2IYRQNvII41NVWhJsarEk8c4qxs/a5sXf";
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,7 +0,0 @@
|
|||||||
{ pkgs, lib, config, ... }: {
|
|
||||||
imports =
|
|
||||||
[
|
|
||||||
# ./home
|
|
||||||
./nixos
|
|
||||||
];
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
{ pkgs, lib, config, ... }: {
|
|
||||||
imports =
|
|
||||||
[
|
|
||||||
./graphical-desktop.nix
|
|
||||||
];
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
{ pkgs, lib, config, ... }: {
|
|
||||||
imports =
|
|
||||||
[
|
|
||||||
./bundles
|
|
||||||
# ./programs
|
|
||||||
./services
|
|
||||||
./filesystem
|
|
||||||
];
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
{ pkgs, lib, config, ... }: {
|
|
||||||
imports =
|
|
||||||
[
|
|
||||||
./hoardingcow-mount.nix
|
|
||||||
];
|
|
||||||
}
|
|
||||||
105
modules/nixos/security/README-ai-worker.md
Normal file
105
modules/nixos/security/README-ai-worker.md
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
# AI Worker Restricted Access
|
||||||
|
|
||||||
|
This module provides SSH access for the AI worker (hermes-agent) to run ollama benchmarks on the host.
|
||||||
|
|
||||||
|
## Security Model
|
||||||
|
|
||||||
|
The `ai-worker` user has:
|
||||||
|
|
||||||
|
### Filesystem Access
|
||||||
|
- **Home directory**: `/home/ai-worker` (standard user home)
|
||||||
|
- **No bind mounts**: Cannot access `/home/gortium/infra` or other host files
|
||||||
|
- **Cannot access**: Any files outside standard system paths
|
||||||
|
|
||||||
|
### Sudo Access
|
||||||
|
- **NONE**: ai-worker has no sudo privileges
|
||||||
|
- Cannot run `nh`, `nixos-rebuild`, `nixpkgs-fmt`, or `nix` with elevated permissions
|
||||||
|
|
||||||
|
### Docker Access
|
||||||
|
- Member of `docker` group - can run `docker` and `docker exec` commands
|
||||||
|
- Primary use: `docker exec ollama ollama ...` for benchmarking
|
||||||
|
- Can run `docker exec --privileged ollama rocm-smi ...` for VRAM monitoring
|
||||||
|
|
||||||
|
## Workflow: SSH + Docker Benchmarking
|
||||||
|
|
||||||
|
The AI worker connects from the Hermes container to the host via SSH, runs ollama benchmarks, then returns to save results.
|
||||||
|
|
||||||
|
### Example Workflow
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# From Hermes container, SSH to host
|
||||||
|
ssh -i /path/to/ssh/key ai-worker@host.docker.internal
|
||||||
|
|
||||||
|
# On host, run ollama benchmarks via docker
|
||||||
|
docker exec ollama ollama pull devstral-small-2:24b
|
||||||
|
|
||||||
|
# Create test modelfile
|
||||||
|
docker exec ollama bash -c 'cat <<EOF > /root/.ollama/test.modelfile
|
||||||
|
FROM devstral-small-2:24b
|
||||||
|
PARAMETER num_ctx 65536
|
||||||
|
PARAMETER num_gpu 99
|
||||||
|
PARAMETER flash_attn true
|
||||||
|
EOF'
|
||||||
|
|
||||||
|
# Create and test model
|
||||||
|
docker exec ollama ollama create test-model -f /root/.ollama/test.modelfile
|
||||||
|
docker exec ollama ollama run test-model "Write a Python async function"
|
||||||
|
|
||||||
|
# Check VRAM usage
|
||||||
|
docker exec --privileged ollama rocm-smi --showmeminfo vram
|
||||||
|
|
||||||
|
# Cleanup
|
||||||
|
docker exec ollama ollama rm test-model
|
||||||
|
|
||||||
|
# Exit SSH, return to Hermes container
|
||||||
|
exit
|
||||||
|
|
||||||
|
# Save results in Hermes container
|
||||||
|
# /opt/data/ai-optimizer/state.json
|
||||||
|
# /opt/data/ai-optimizer/results.csv
|
||||||
|
```
|
||||||
|
|
||||||
|
## SSH Access
|
||||||
|
|
||||||
|
Connect as:
|
||||||
|
```bash
|
||||||
|
ssh ai-worker@lazyworkhorse
|
||||||
|
```
|
||||||
|
|
||||||
|
The working directory will be `/home/ai-worker`. No infra repo access.
|
||||||
|
|
||||||
|
## Verification
|
||||||
|
|
||||||
|
Check ai-worker permissions:
|
||||||
|
```bash
|
||||||
|
# On the host, as root or gortium:
|
||||||
|
sudo -u ai-worker sudo -l
|
||||||
|
# Should show: no sudo access
|
||||||
|
|
||||||
|
# Check docker group membership
|
||||||
|
groups ai-worker
|
||||||
|
# Should show: ai-worker docker
|
||||||
|
```
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
If ai-worker cannot run docker commands:
|
||||||
|
```bash
|
||||||
|
# Check docker group membership
|
||||||
|
groups ai-worker
|
||||||
|
|
||||||
|
# Verify ollama container is running
|
||||||
|
docker ps | grep ollama
|
||||||
|
|
||||||
|
# Test docker access
|
||||||
|
sudo -u ai-worker docker exec ollama ollama list
|
||||||
|
```
|
||||||
|
|
||||||
|
If SSH connection fails:
|
||||||
|
```bash
|
||||||
|
# Check SSH key is authorized
|
||||||
|
cat /home/ai-worker/.ssh/authorized_keys
|
||||||
|
|
||||||
|
# Check SSH service
|
||||||
|
systemctl status sshd
|
||||||
|
```
|
||||||
17
modules/nixos/security/ai-worker-restricted.nix
Normal file
17
modules/nixos/security/ai-worker-restricted.nix
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
{ config, pkgs, lib, ... }:
|
||||||
|
|
||||||
|
with lib;
|
||||||
|
|
||||||
|
{
|
||||||
|
options.services.aiWorkerAccess = mkOption {
|
||||||
|
type = types.bool;
|
||||||
|
default = false;
|
||||||
|
description = "Enable AI worker SSH access with docker group membership for ollama benchmarking";
|
||||||
|
};
|
||||||
|
|
||||||
|
config = mkIf config.services.aiWorkerAccess {
|
||||||
|
# ai-worker is member of docker group - can run docker commands via SSH
|
||||||
|
# No bind mounts, no sudo access - docker-only for ollama benchmarking
|
||||||
|
users.groups.docker.members = [ "ai-worker" ];
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
{ pkgs, lib, config, ... }: {
|
|
||||||
imports =
|
|
||||||
[
|
|
||||||
./systemd
|
|
||||||
];
|
|
||||||
}
|
|
||||||
@@ -9,9 +9,15 @@ with lib;
|
|||||||
path = mkOption { type = types.str; };
|
path = mkOption { type = types.str; };
|
||||||
envFile = mkOption { type = types.nullOr types.path; default = null; };
|
envFile = mkOption { type = types.nullOr types.path; default = null; };
|
||||||
ports = mkOption { type = types.listOf types.int; default = [ ]; };
|
ports = mkOption { type = types.listOf types.int; default = [ ]; };
|
||||||
|
# New option to pass raw systemd serviceConfig
|
||||||
|
serviceConfig = mkOption {
|
||||||
|
type = types.attrs;
|
||||||
|
default = { };
|
||||||
|
description = "Extra systemd serviceConfig options for this stack.";
|
||||||
|
};
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
default = {};
|
default = { };
|
||||||
};
|
};
|
||||||
|
|
||||||
config = {
|
config = {
|
||||||
@@ -23,28 +29,29 @@ with lib;
|
|||||||
systemd.services = mapAttrs' (name: value: nameValuePair "${name}_stack" {
|
systemd.services = mapAttrs' (name: value: nameValuePair "${name}_stack" {
|
||||||
description = "Docker Compose stack: ${name}";
|
description = "Docker Compose stack: ${name}";
|
||||||
|
|
||||||
# Added 'docker.socket' to both after and wants to ensure the API is reachable
|
# Forces systemd to restart when the files change
|
||||||
|
reloadTriggers = [
|
||||||
|
"${builtins.hashFile "sha256" (toString value.path + "/compose.yml")}"
|
||||||
|
] ++ (lib.optional (value.envFile != null) "${value.envFile}");
|
||||||
|
|
||||||
after = [ "network.target" "docker.service" "docker.socket" "agenix.service" ];
|
after = [ "network.target" "docker.service" "docker.socket" "agenix.service" ];
|
||||||
wants = [ "docker.socket" "agenix.service" ];
|
wants = [ "docker.socket" "agenix.service" ];
|
||||||
requires = [ "docker.service" ];
|
requires = [ "docker.service" ];
|
||||||
|
|
||||||
wantedBy = [ "multi-user.target" ];
|
wantedBy = [ "multi-user.target" ];
|
||||||
|
|
||||||
serviceConfig = {
|
path = with pkgs; [ git docker docker-compose bash ];
|
||||||
|
|
||||||
|
# We merge the base config with the custom 'serviceConfig' from the submodule
|
||||||
|
serviceConfig = recursiveUpdate {
|
||||||
Type = "oneshot";
|
Type = "oneshot";
|
||||||
WorkingDirectory = value.path;
|
WorkingDirectory = value.path;
|
||||||
User = "root";
|
User = "root";
|
||||||
|
|
||||||
# This line forces the service to wait until the docker socket is actually responsive
|
|
||||||
ExecStartPre = "${pkgs.bash}/bin/bash -c 'while [ ! -S /var/run/docker.sock ]; do sleep 1; done'";
|
ExecStartPre = "${pkgs.bash}/bin/bash -c 'while [ ! -S /var/run/docker.sock ]; do sleep 1; done'";
|
||||||
|
|
||||||
ExecStart = "${pkgs.docker-compose}/bin/docker-compose up -d --remove-orphans";
|
ExecStart = "${pkgs.docker-compose}/bin/docker-compose up -d --remove-orphans";
|
||||||
ExecStop = "${pkgs.docker-compose}/bin/docker-compose down";
|
ExecStop = "${pkgs.docker-compose}/bin/docker-compose down";
|
||||||
RemainAfterExit = true;
|
RemainAfterExit = true;
|
||||||
|
|
||||||
# Ensure the environment file is passed correctly
|
|
||||||
EnvironmentFile = mkIf (value.envFile != null) [ value.envFile ];
|
EnvironmentFile = mkIf (value.envFile != null) [ value.envFile ];
|
||||||
};
|
} value.serviceConfig;
|
||||||
}) config.services.dockerStacks;
|
}) config.services.dockerStacks;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,45 +1,87 @@
|
|||||||
{ pkgs, ... }: {
|
{ pkgs, ... }: {
|
||||||
systemd.services.init-ollama-model = {
|
systemd.services.init-ollama-model = {
|
||||||
description = "Initialize LLM models with extra context in Ollama Docker";
|
description = "Initialize LLM models with extra context in Ollama Docker";
|
||||||
after = [ "docker-ollama.service" ];
|
|
||||||
|
# On s'assure que Docker tourne avant de lancer ce script
|
||||||
|
after = [ "docker.service" ];
|
||||||
wantedBy = [ "multi-user.target" ];
|
wantedBy = [ "multi-user.target" ];
|
||||||
|
|
||||||
script = ''
|
script = ''
|
||||||
# Wait for Ollama
|
# Fonction de création asynchrone pour ne pas bloquer le démarrage
|
||||||
while ! ${pkgs.curl}/bin/curl -s http://localhost:11434/api/tags > /dev/null; do
|
(
|
||||||
sleep 2
|
echo "Starting asynchronous Ollama initialization..."
|
||||||
done
|
|
||||||
|
# Attente d'Ollama (maximum 120 secondes pour éviter une boucle infinie)
|
||||||
|
TIMEOUT=60
|
||||||
|
COUNT=0
|
||||||
|
while ! ${pkgs.curl}/bin/curl -s -f http://127.0.0.1:11434/api/tags > /dev/null; do
|
||||||
|
if [ $COUNT -ge $TIMEOUT ]; then
|
||||||
|
echo "Ollama did not become ready in time. Exiting."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "Waiting for Ollama API to be reachable..."
|
||||||
|
sleep 5
|
||||||
|
COUNT=$((COUNT + 5))
|
||||||
|
done
|
||||||
|
|
||||||
create_model_if_missing() {
|
create_model_if_missing() {
|
||||||
local model_name=$1
|
local model_name=$1
|
||||||
local base_model=$2
|
local base_model=$2
|
||||||
if ! ${pkgs.docker}/bin/docker exec ollama ollama list | grep -q "$model_name"; then
|
|
||||||
echo "$model_name not found, creating from $base_model..."
|
# Vérification robuste via l'API HTTP d'Ollama plutôt que docker exec (évite les conflits de tty)
|
||||||
${pkgs.docker}/bin/docker exec ollama sh -c "cat <<EOF > /root/.ollama/$model_name.modelfile
|
if ! ${pkgs.curl}/bin/curl -s http://127.0.0.1:11434/api/tags | ${pkgs.jq}/bin/jq -e ".models[] | select(.name == \"$model_name\")" > /dev/null; then
|
||||||
|
echo "$model_name not found, creating from $base_model..."
|
||||||
|
|
||||||
|
# Utilisation d'un fichier temporaire sur l'hôte pour l'injecter proprement dans Docker
|
||||||
|
TMP_FILE=$(mktemp)
|
||||||
|
cat <<EOF > "$TMP_FILE"
|
||||||
FROM $base_model
|
FROM $base_model
|
||||||
|
TEMPLATE """{{- if .System }}
|
||||||
|
[SYSTEM_PROMPT]
|
||||||
|
{{ .System }}
|
||||||
|
[/SYSTEM_PROMPT]
|
||||||
|
{{- end }}
|
||||||
|
{{- range .Messages }}
|
||||||
|
{{- if eq .Role "user" }}
|
||||||
|
[INST]
|
||||||
|
{{ .Content }}
|
||||||
|
[/INST]
|
||||||
|
{{- else if eq .Role "assistant" }}
|
||||||
|
{{ .Content }}
|
||||||
|
{{- end }}
|
||||||
|
{{- end }}"""
|
||||||
PARAMETER num_ctx 131072
|
PARAMETER num_ctx 131072
|
||||||
PARAMETER num_predict 4096
|
PARAMETER num_predict 4096
|
||||||
PARAMETER num_keep 1024
|
PARAMETER num_keep 1024
|
||||||
PARAMETER repeat_penalty 1.1
|
PARAMETER repeat_penalty 1.1
|
||||||
PARAMETER top_k 40
|
PARAMETER top_k 40
|
||||||
PARAMETER stop \"[INST]\"
|
PARAMETER stop "[INST]"
|
||||||
PARAMETER stop \"[/INST]\"
|
PARAMETER stop "[/INST]"
|
||||||
PARAMETER stop \"</s>\"
|
PARAMETER stop "</s>"
|
||||||
EOF"
|
EOF
|
||||||
${pkgs.docker}/bin/docker exec ollama ollama create "$model_name" -f "/root/.ollama/$model_name.modelfile"
|
|
||||||
else
|
|
||||||
echo "$model_name already exists, skipping."
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# Create Nemotron
|
# Copie et création dans le conteneur
|
||||||
create_model_if_missing "nemotron-3-nano:30b-128k" "nemotron-3-nano:30b"
|
${pkgs.docker}/bin/docker cp "$TMP_FILE" ollama:/tmp/model.modelfile
|
||||||
|
${pkgs.docker}/bin/docker exec ollama ollama create "$model_name" -f /tmp/model.modelfile
|
||||||
# Create Devstral
|
${pkgs.docker}/bin/docker exec ollama rm /tmp/model.modelfile
|
||||||
create_model_if_missing "devstral-small-2:24b-128k" "devstral-small-2:24b"
|
rm -f "$TMP_FILE"
|
||||||
|
else
|
||||||
|
echo "$model_name already exists, skipping."
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# Create Nemotron
|
||||||
|
create_model_if_missing "nemotron-3-nano:30b-128k" "nemotron-3-nano:30b"
|
||||||
|
|
||||||
|
# Create Devstral
|
||||||
|
create_model_if_missing "devstral-small-2:24b-128k" "devstral-small-2:24b"
|
||||||
|
|
||||||
|
) &
|
||||||
'';
|
'';
|
||||||
|
|
||||||
serviceConfig = {
|
serviceConfig = {
|
||||||
Type = "oneshot";
|
Type = "forking"; # Permet à systemd de savoir que le script passe en arrière-plan via '&'
|
||||||
RemainAfterExit = true;
|
User = "root";
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,11 +20,7 @@ in {
|
|||||||
|
|
||||||
environment.etc."opencode/opencode.json".text = builtins.toJSON {
|
environment.etc."opencode/opencode.json".text = builtins.toJSON {
|
||||||
"$schema" = "https://opencode.ai/config.json";
|
"$schema" = "https://opencode.ai/config.json";
|
||||||
"model" = "devstral-2-small-llama_cpp";
|
"model" = "nemotron-3-nano-llama_cpp";
|
||||||
|
|
||||||
# MCP servers for web search and enhanced functionality
|
|
||||||
# context7: Remote HTTP server for up-to-date documentation and code examples
|
|
||||||
# duckduckgo: Local MCP server for web search capabilities
|
|
||||||
"mcp" = {
|
"mcp" = {
|
||||||
"context7" = {
|
"context7" = {
|
||||||
"type" = "remote";
|
"type" = "remote";
|
||||||
@@ -46,6 +42,7 @@ in {
|
|||||||
"options" = {
|
"options" = {
|
||||||
"baseURL" = "http://localhost:8300/v1";
|
"baseURL" = "http://localhost:8300/v1";
|
||||||
"apiKey" = "not-needed";
|
"apiKey" = "not-needed";
|
||||||
|
"maxTokens" = 80000;
|
||||||
};
|
};
|
||||||
"models" = {
|
"models" = {
|
||||||
"devstral-2-small-llama_cpp" = {
|
"devstral-2-small-llama_cpp" = {
|
||||||
@@ -53,6 +50,11 @@ in {
|
|||||||
"tools" = true;
|
"tools" = true;
|
||||||
"reasoning" = false;
|
"reasoning" = false;
|
||||||
};
|
};
|
||||||
|
"nemotron-3-nano-llama_cpp" = {
|
||||||
|
"name" = "Nemotron 3 nano 30B Q8 (llama.cpp)";
|
||||||
|
"tools" = true;
|
||||||
|
"reasoning" = false;
|
||||||
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
"ollama" = {
|
"ollama" = {
|
||||||
@@ -76,6 +78,7 @@ in {
|
|||||||
systemd.services.opencode-gsd-install = {
|
systemd.services.opencode-gsd-install = {
|
||||||
description = "Install Get Shit Done OpenCode Components";
|
description = "Install Get Shit Done OpenCode Components";
|
||||||
after = [ "network-online.target" ];
|
after = [ "network-online.target" ];
|
||||||
|
wants = [ "network-online.target" ];
|
||||||
wantedBy = [ "multi-user.target" ];
|
wantedBy = [ "multi-user.target" ];
|
||||||
path = with pkgs; [
|
path = with pkgs; [
|
||||||
nodejs
|
nodejs
|
||||||
@@ -131,7 +134,6 @@ in {
|
|||||||
|
|
||||||
environment = {
|
environment = {
|
||||||
OLLAMA_BASE_URL = "http://127.0.0.1:11434";
|
OLLAMA_BASE_URL = "http://127.0.0.1:11434";
|
||||||
# Important: GSD at ~/.config/opencode, so we ensure the server sees our /etc config
|
|
||||||
OPENCODE_CONFIG = "/etc/opencode/opencode.json";
|
OPENCODE_CONFIG = "/etc/opencode/opencode.json";
|
||||||
HOME = "/home/gortium";
|
HOME = "/home/gortium";
|
||||||
NODE_PATH = "${pkgs.nodejs}/lib/node_modules";
|
NODE_PATH = "${pkgs.nodejs}/lib/node_modules";
|
||||||
|
|||||||
64
modules/nixos/services/openclaw_node.nix
Normal file
64
modules/nixos/services/openclaw_node.nix
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
{ config, lib, pkgs, ... }:
|
||||||
|
|
||||||
|
let
|
||||||
|
cfg = config.services.openclaw-node;
|
||||||
|
openclawPkg = pkgs.openclaw;
|
||||||
|
in {
|
||||||
|
options.services.openclaw-node = {
|
||||||
|
enable = lib.mkEnableOption "OpenClaw Node service";
|
||||||
|
|
||||||
|
user = lib.mkOption {
|
||||||
|
type = lib.types.str;
|
||||||
|
default = "ai-worker";
|
||||||
|
description = "User to run the OpenClaw headless node as.";
|
||||||
|
};
|
||||||
|
|
||||||
|
gatewayHost = lib.mkOption {
|
||||||
|
type = lib.types.str;
|
||||||
|
default = "127.0.0.1";
|
||||||
|
description = "Gateway host (IP or hostname).";
|
||||||
|
};
|
||||||
|
|
||||||
|
gatewayPort = lib.mkOption {
|
||||||
|
type = lib.types.int;
|
||||||
|
default = 18789;
|
||||||
|
description = "Gateway WebSocket port.";
|
||||||
|
};
|
||||||
|
|
||||||
|
gatewayTokenFile = lib.mkOption {
|
||||||
|
type = lib.types.str;
|
||||||
|
default = "";
|
||||||
|
description = "Path to file containing the gateway auth token.";
|
||||||
|
};
|
||||||
|
|
||||||
|
displayName = lib.mkOption {
|
||||||
|
type = lib.types.str;
|
||||||
|
default = "lazyworkhorse-host";
|
||||||
|
description = "Display name for this node (shown in pairing).";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
config = lib.mkIf cfg.enable {
|
||||||
|
systemd.services.openclaw-node = {
|
||||||
|
description = "OpenClaw Headless Node Service";
|
||||||
|
after = [ "network.target" ];
|
||||||
|
wantedBy = [ "multi-user.target" ];
|
||||||
|
|
||||||
|
serviceConfig = {
|
||||||
|
Type = "exec";
|
||||||
|
User = cfg.user;
|
||||||
|
Group = cfg.user;
|
||||||
|
WorkingDirectory = "/home/${cfg.user}";
|
||||||
|
ExecStart = ''
|
||||||
|
${pkgs.bash}/bin/bash -c 'export OPENCLAW_GATEWAY_TOKEN=$(cat ${cfg.gatewayTokenFile}) && exec ${openclawPkg}/bin/openclaw node run --host ${cfg.gatewayHost} --port ${toString cfg.gatewayPort} --display-name "${cfg.displayName}"'
|
||||||
|
'';
|
||||||
|
Restart = "always";
|
||||||
|
RestartSec = 5;
|
||||||
|
};
|
||||||
|
|
||||||
|
environment = {
|
||||||
|
NODE_ENV = "production";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -1,24 +1,36 @@
|
|||||||
-----BEGIN AGE ENCRYPTED FILE-----
|
-----BEGIN AGE ENCRYPTED FILE-----
|
||||||
YWdlLWVuY3J5cHRpb24ub3JnL3YxCi0+IHNzaC1lZDI1NTE5IEdoTUQ4QSBGWmpW
|
YWdlLWVuY3J5cHRpb24ub3JnL3YxCi0+IHNzaC1lZDI1NTE5IEdoTUQ4QSBMcVY2
|
||||||
bFFuT1FNWVlsd0twcUJnYXV0T0Z3Q0RDZldsNTUwWlprQTJaK2xNCmMzS3g1OEdI
|
eWRnb05sMEZVZUpicGVYN2ZSUm1mUmVsdUsrSHR4c2I2SXVRWDF3CnlhUkxoeEx5
|
||||||
bENzekRFTkIwbVRua2MzTVdnZmNKMnd6dzJjZEx5eXhBWmMKLT4gV2ktZ3JlYXNl
|
M1NNcGR3bmx6ZW5RaHU3L2l4WEowdWYzQk0xa3E4ZUtwNkkKLT4gIi1ncmVhc2Ug
|
||||||
IChaQl14QSB3IFlIcmkKVHZPSmZ2aXNaSHVUbi9UbUNTL00ycWRZbzVwTlFUUjls
|
SllwSzlIIEdAKltUKSBjSWpmCnljSXlZTXZBL2xuMEtma1NUNjdCM0dMNHJuVjFS
|
||||||
Z2RFSGMyM2ZDbkRlekxxemR4RTlLWnI3L0FlanpkYgpaaUlpSFdxZlo0Sk9XcXF3
|
enV5LzF1NlYzbFNURW1rTlI2aUxCRFFxK2ZJdktsTTU1ZSsKNUZiZmVQWQotLS0g
|
||||||
TnZQYzY1MWxLRklycWh3MEl2ZENSMk5yMDNKNWkyZmVBNTlSNWxBSzZ2RDNmeDRP
|
bTVrZEdWMHFWK0Q5RzZUc09PYmlEQjNweXRxU2FETWNWTXRUVEFKUUFpdwqtUbmM
|
||||||
CgotLS0gNEtpRlhJbkZXcGNpQzBFREhCempyYlFHcTRHSlpTOUZFeGxmNHk2c20x
|
kbnj4Q+9QlnJHuWBVu+BcWPl5HuiPUjrrxnWAuN6rFYd79H7qk9MagDB/2BAEk82
|
||||||
VQrxqxWUB/GZUQixOXxdZhfeUDyzbc7DZ4CMA8o0X0NHxxonsHQXvAwcHFYVBj45
|
iuQeqS0r8wAmp9bTzhbiMQEbtN96huSGA2aO9I/RoPU1jv1Upi8bNy+KX0jSsfV9
|
||||||
d7D9yjtHYP+EAR2skUEnlPYfUdFKtjyE4KRE/wv6VQXfjeIax0USypvuEg9e+cfA
|
+cGFm0JsBZ4o5acq4isanC+3YkNq+IHxyDqUdwWMIHiEkR26pwndyMAaJTCBAgTN
|
||||||
VknSLO4G+si8MvccJNZsBGGebEg8OpmSqSog6pee3jeVtr0fr5no0901rnwZYQEN
|
kmdWMK4PxR36ElY+8J0tiOv6VmGS73rVS4R/2Rdc5ICx6QO6oQQqETKYxqcmr8eD
|
||||||
X63i+8cp2ZnHCxuR6ol48rUB9AEieYiYvI8gCfATigvFkjj/fEYKLK/kgqLVl96p
|
8dgkZVpmBF/iuw7BYZ2U/p2PZSzgEVTVqfT3VO8WQ9ii2a4dw+2QEORgISbhxzoy
|
||||||
CjtXqhO0XGROPCvyVB8yadJCw67tMdkZO39saJTeHP6r0lz37lHNm8Uwyel89kLd
|
WGB0q4X33QWEYSp0FNyAG4Kc9yTphUq+hMHfNOtz3XmTZLltcVfA6XQbKwB/nDbq
|
||||||
CWqrIK67MH1ejXwhTfQlHSX3WQYAXfxq7fmetjcJb0NBXUBsPrAwlmz49T0TWvfa
|
G1gAfDwdEZ5OpN2IT0S6Zc2rKKeovqdFYWgqmDWiaBfqncl9qLH37KkpKqmUSQSe
|
||||||
1oi60xLD+BsKR3KDgthid3GwhcrsY5RA8y8x8c4Ssk1iLKEIlyOM+f2cYJRvYMrS
|
zyriQt8nZCzVrh4EKohjeLogVBsn0tPTYqiFnV+ZFK/kOWIYWduWDJcM3zwRVjx7
|
||||||
LfSs1cvIORLA8QcADELhzV7mVsBtXo8vU5oSoCWrvT0vs2H2EFvl4Qfx/8UGoVMK
|
Mr5l81gFv4WRHbnQB9eynGJLZYs1lI4/X9tKRUgUj0mN20Bt80NXGNPaJ3TAkrCO
|
||||||
p3HFMw3Qwxh2Qyr6kD6SuRc1dzbseXiBtPuN76KOQNbo9LEu0JNwsoHqv7wdUS6u
|
rX0tgjsX/Sc59JTHaMOT419+ob9UtDoS7mGlxswKfyhvjzluu+EdHd0GRJ5PUsSq
|
||||||
r831UKyTxWfl3oBUzldG2Ugka3/7wr3n2biARkADNjrvkFHo5BM6vYla583j6ml3
|
8YlKYjRonBqhC0Ju9CRuZS0pk2bh72bG9z1Gb4LcJ0pLJ6lMfmF1j4zzfzsABe5G
|
||||||
/IzQOIQXSmgv+opza1oghf2jg9UFkMOPZ9iz6srg2xaH+xZ7+xnL3cuY4ngWwIqy
|
0bJF3ikzNxo6Wzsf3rk45/FvMhKjkI5O3Btnfp+Vkw+iRqDMh7wiD6cczNSp+Skd
|
||||||
pRKdcrNDOIawhEpJEAUYLHMcrCCekZPJalEcMZ26pXjVG1p9SYVsQWxkpVgOqEIH
|
Et43NIobiLa6O95y2YEjqSkT5T0ug1nbLkygmxwffVn6RTWgScZSfBPvOKTINAVo
|
||||||
8Q4zYMYQAQssVSED3SrQ39giW7+UfGnoqsy9qTq1UvDBpnGDMk2JYsGZmQoWEvtJ
|
J/MU2c0DaBm4glLfm4IWaJNcEmZn8+FWG6m42WEWTMEfSeAo5XXaEb0FsK0Yqd3+
|
||||||
AudwoHTFj/szABXE7qootqjGGhopdC0pFWGKaSFRre7iIeiYNJDXYi1lyAtDfZFW
|
RlE/b6a/DdoNEAQOlEPSASsoQhTVvsiEwH4Pq5G0STHtSBBIT/xJow4pa/GOiQnn
|
||||||
iv8avbywunozAigA8+wuF4Zw1GOThPAOLNU=
|
yAFva9F7KFYWA7bjbRbv+B21bvss0T+BK9HRF+bklSzjxBNfDEWXW0GhwIiahwXW
|
||||||
|
Fxi3BUMZcfgoLg2NkhMb3irwJUoGqQFqn68e2jTJVyUATyVgGV5mzfjDBB8thcDt
|
||||||
|
ehuIL/Y1bUVGWZteU/JAF7z+Fb1yBO2FJqpCKIcfd+JgkWVtQKaqYGjcuzbI7ce4
|
||||||
|
fcrRDNCse2pBO1unE7HS160rK+dMWavlrsHHZKezsvNv7TwMj1SKjCKSVr8up7TC
|
||||||
|
NLW0I6uXGEcUBj+RNqF6frdpw/Ve1WTAkIbznMV6kZ8cTfAGhOzJ0wxHMBSQka8X
|
||||||
|
uMatPuGywu+dvjrJXTwD0gJD/Jrd88K685ahu86nSt/DYnxIYfQhwo/oZMR7E6kN
|
||||||
|
4NgGtYjRU0asmio6sF8D1uA2YgmxNPRw2GwTqpS2XyYaEh3B6yjqa4pJ1vfo6t/g
|
||||||
|
aPLhMuT4qt/eKMvSiR+sTaOMNkcLWmCpY762aYk4XUZFTYAAY5XsNSAU+Hs7o/Eu
|
||||||
|
9NSMrIrpqLAcgzr/nZZTLqswbsYtdVl5WUd9uqdQe+AhTbrkcvHibUrVgW/XU+oq
|
||||||
|
QdHXUyfn/IByp2uE7WZpfRVhwjXg/LQJh/ogksbzrh4/anOivku+n1ouciAAgnQn
|
||||||
|
04i6taBu+393ysMC/sp7wZkp//yAZj2SNPOTzbakG8xWoVGzbqxLCIOIE7I97KBv
|
||||||
|
G49jiEOS42vZZXSGLxQND+N0aOqosZfQ1WKpI9XjirB8qVOt/sr6uqEIx311V+BJ
|
||||||
|
wrdoMa9hWmDFzf3+ThqfUuHOtxxJXReL7vC4J7K8iU6nCVIGJN7axifk
|
||||||
-----END AGE ENCRYPTED FILE-----
|
-----END AGE ENCRYPTED FILE-----
|
||||||
|
|||||||
11
secrets/openclaw_gateway_token.age
Normal file
11
secrets/openclaw_gateway_token.age
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
-----BEGIN AGE ENCRYPTED FILE-----
|
||||||
|
YWdlLWVuY3J5cHRpb24ub3JnL3YxCi0+IHNzaC1lZDI1NTE5IEdoTUQ4QSBCWEpO
|
||||||
|
cG9yNnFpcHFqTkNzTngxU1MxN0NYK0hrZFhUTjVORWFrK3JNd2tZCmtMTGpwQk1E
|
||||||
|
WlUwL3N6SGRWblpnNEkrWkkyU2hQMkRIK0M3R0pOVEREV3MKLT4gY2osLWdyZWFz
|
||||||
|
ZSBacSozVVQgUCAxRS1OQSAuKXxDPCoKbStWNW1BZjBZQzNDaTlDbU5EZkxsRWxM
|
||||||
|
cXJ3dDU1RDNpOXRlV0tzdEp2NUo3S1lhRG5Md0RHTGlJdkFSYmt5YQo4R1hiQWRG
|
||||||
|
V2VxekJKZwotLS0geG1XSi9VbkhXZHQzcEFVS3hKNzVueXFLa2xnZTc3Q2tJTVZ5
|
||||||
|
eXJabWk5Ywp6bJCP3s0xxzjE+eTR+cv7ZUnkoliT/n7uIprq1BTn/LIRLkUTUqs3
|
||||||
|
NiDwrXcoq4/QKd0Dt+8ap3vFAuusjGxRlnYMaRrZie2AGtTV8U7Q7durm9o2K+/4
|
||||||
|
QzRQ/MtumIQm
|
||||||
|
-----END AGE ENCRYPTED FILE-----
|
||||||
@@ -10,4 +10,5 @@ in
|
|||||||
"containers.env.age".publicKeys = authorizedKeys;
|
"containers.env.age".publicKeys = authorizedKeys;
|
||||||
"lazyworkhorse_host_ssh_key.age".publicKeys = authorizedKeys;
|
"lazyworkhorse_host_ssh_key.age".publicKeys = authorizedKeys;
|
||||||
"n8n_ssh_key.age".publicKeys = authorizedKeys;
|
"n8n_ssh_key.age".publicKeys = authorizedKeys;
|
||||||
|
"openclaw_gateway_token.age".publicKeys = authorizedKeys;
|
||||||
}
|
}
|
||||||
|
|||||||
9
secrets/wireguard_preshared_key.age
Normal file
9
secrets/wireguard_preshared_key.age
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
-----BEGIN AGE ENCRYPTED FILE-----
|
||||||
|
YWdlLWVuY3J5cHRpb24ub3JnL3YxCi0+IHNzaC1lZDI1NTE5IEdoTUQ4QSA3VG9Z
|
||||||
|
MVFPVFc2VVJ3d0h0dmtBUnI3WHl2SzUxTkRZbjFCaGloWmV3dnd3ClcxdnVPeGd6
|
||||||
|
SU4zR0Q0K1dtVjRRVHd0VW5XSFI0dVFpTjZnYk1DNjRxTVEKLT4gQzlgRy1ncmVh
|
||||||
|
c2UKeUozOWgyUytSTVF0NjY2STBEb2VadwotLS0gblI3bmJCUWxxU3QrYTEyVFBI
|
||||||
|
Snc4NC9rTkh0NnZYbUtxUE9hRWRkelpmMAq58fmH6cK13GeD7wGLxKmx10hmJeW4
|
||||||
|
b7KqnCD1ZP7uG85s32xzVRwRG8RrG4xZo5nR9Mrtg1CoTSFfUGeFnf5xveN+Ej0X
|
||||||
|
wDVB1LwC+Q==
|
||||||
|
-----END AGE ENCRYPTED FILE-----
|
||||||
11
secrets/wireguard_private_key.age
Normal file
11
secrets/wireguard_private_key.age
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
-----BEGIN AGE ENCRYPTED FILE-----
|
||||||
|
YWdlLWVuY3J5cHRpb24ub3JnL3YxCi0+IHNzaC1lZDI1NTE5IEdoTUQ4QSA5dzVG
|
||||||
|
WUNvT3NlRmcrWS81bzJqSWlTekVYaDFFTE10SkI2dEgzaGpxcUI4Cmk5Y0FGYTRZ
|
||||||
|
K0NGYzY3VUp4aS9ZZGRmWTgybDJFUURva2pZNmVOS3QxdEUKLT4gPnVRTCtldGMt
|
||||||
|
Z3JlYXNlCk04OTJZeFRNeDI5aGpMVTk1ZTE0Y2FMMnFEMjlJalJpMHRlaTE4ZWIx
|
||||||
|
d2lCRGQ5RHVjcktOMGJCb1VERlNWcTYKaSt0L1Z6dVJ0QWIyZkhsYzFEVjZSQWUr
|
||||||
|
ZWpwVlo1TmhoUFJZdkEvR0gxNlVhcXF2ZTRnCi0tLSBLcmM2MThNVkdWclpHUXRr
|
||||||
|
VTF6QVk2WUZlTXpZMVNLMlpBOFc3M1o5WjZzCs9xbPlIX+u5vRSQ/z9utu+I9S2c
|
||||||
|
02DOsIb1kzxzb1OK91b8Kh4JucQSq3qkyEvRucsNn5QW8hIHDnRuND6EbPyN7p4S
|
||||||
|
YB/F0dxSqgnq
|
||||||
|
-----END AGE ENCRYPTED FILE-----
|
||||||
93
users/ai-worker.nix
Normal file
93
users/ai-worker.nix
Normal file
@@ -0,0 +1,93 @@
|
|||||||
|
{ pkgs, inputs, config, keys, ... }: {
|
||||||
|
users.users.ai-worker = {
|
||||||
|
isSystemUser = true;
|
||||||
|
group = "ai-worker";
|
||||||
|
home = "/home/ai-worker";
|
||||||
|
createHome = true;
|
||||||
|
extraGroups = [ "docker" ];
|
||||||
|
shell = pkgs.bashInteractive;
|
||||||
|
openssh.authorizedKeys.keys = [
|
||||||
|
keys.users.ai-worker.main
|
||||||
|
];
|
||||||
|
# No password login - SSH key only
|
||||||
|
hashedPassword = "!";
|
||||||
|
};
|
||||||
|
users.groups.ai-worker = {};
|
||||||
|
|
||||||
|
# Enable restricted AI worker SSH access for ollama benchmarking
|
||||||
|
# SECURITY: ai-worker can only:
|
||||||
|
# - SSH into host from Hermes container
|
||||||
|
# - Run docker commands (docker exec ollama ...) via docker group
|
||||||
|
# - Run specific security audit commands
|
||||||
|
# - NO access to infra repo (no bind mount)
|
||||||
|
# - NO sudo access (no nh, nixos-rebuild, nixpkgs-fmt, nix)
|
||||||
|
# WORKFLOW: SSH from Hermes container, run docker benchmarks, return and save results to /opt/data/ai-optimizer/
|
||||||
|
services.aiWorkerAccess = true;
|
||||||
|
|
||||||
|
# Restricted sudo for ai-worker - security checks only
|
||||||
|
security.sudo.extraRules = [
|
||||||
|
{
|
||||||
|
users = [ "ai-worker" ];
|
||||||
|
commands = [
|
||||||
|
# Firewall checks
|
||||||
|
{
|
||||||
|
command = "/run/wrappers/bin/sudo iptables -L -n -v";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
command = "/run/wrappers/bin/sudo iptables -S";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
# Fail2ban status
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/fail2ban-client status";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/fail2ban-client status *";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/fail2ban-client get * banned";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
# Log inspection
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/journalctl -t kernel -n 100";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/journalctl -u fail2ban -n 50";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/journalctl -u firewall -n 50";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
# SSH config verification
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/sshd -T";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
# Docker service checks
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/docker ps";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/docker inspect *";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
# Network diagnostics
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/ss -tlnp";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
command = "/run/current-system/sw/bin/cat /proc/net/tcp";
|
||||||
|
options = [ "NOPASSWD" ];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
{ pkgs, inputs, config, keys, ... }: {
|
|
||||||
users.users.n8n-worker = {
|
|
||||||
isSystemUser = true;
|
|
||||||
group = "n8n-worker";
|
|
||||||
extraGroups = [ "docker" ];
|
|
||||||
shell = pkgs.bashInteractive;
|
|
||||||
openssh.authorizedKeys.keys = [
|
|
||||||
keys.users.n8n-worker.main
|
|
||||||
];
|
|
||||||
};
|
|
||||||
users.groups.n8n-worker = {};
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user