{"id":42102872,"url":"https://github.com/getnora-io/nora","last_synced_at":"2026-04-28T01:02:23.560Z","repository":{"id":334579546,"uuid":"1141898045","full_name":"getnora-io/nora","owner":"getnora-io","description":"Lightweight multi-format artifact registry. 13 formats: Docker, Maven, npm, PyPI, Cargo, Go, RubyGems, Terraform, NuGet, and more. Single binary, zero dependencies, \u003c100MB RAM.","archived":false,"fork":false,"pushed_at":"2026-04-27T10:50:58.000Z","size":3780,"stargazers_count":131,"open_issues_count":7,"forks_count":7,"subscribers_count":4,"default_branch":"main","last_synced_at":"2026-04-27T11:18:55.044Z","etag":null,"topics":["air-gapped","artifact-registry","artifactory-alternative","cargo-registry","conan","curation","docker-registry","maven-repository","nexus-alternative","npm-registry","nuget","oci-registry","pypi-registry","registry-proxy","rubygems","rust","self-hosted","single-binary","supply-chain-security","terraform-registry"],"latest_commit_sha":null,"homepage":"https://getnora.dev","language":"Rust","has_issues":true,"has_wiki":null,"has_pages":null,"mirror_url":null,"source_name":null,"license":"mit","status":null,"scm":"git","pull_requests_enabled":true,"icon_url":"https://github.com/getnora-io.png","metadata":{"files":{"readme":"README.md","changelog":"CHANGELOG.md","contributing":"CONTRIBUTING.md","funding":null,"license":"LICENSE","code_of_conduct":"CODE_OF_CONDUCT.md","threat_model":null,"audit":null,"citation":null,"codeowners":".github/CODEOWNERS","security":"SECURITY.md","support":null,"governance":null,"roadmap":null,"authors":null,"dei":null,"publiccode":null,"codemeta":null,"zenodo":null,"notice":null,"maintainers":null,"copyright":null,"agents":null,"dco":null,"cla":null}},"created_at":"2026-01-25T16:13:37.000Z","updated_at":"2026-04-27T10:51:03.000Z","dependencies_parsed_at":"2026-02-01T13:01:15.080Z","dependency_job_id":null,"html_url":"https://github.com/getnora-io/nora","commit_stats":null,"previous_names":["getnora-io/nora"],"tags_count":44,"template":false,"template_full_name":null,"purl":"pkg:github/getnora-io/nora","repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/getnora-io%2Fnora","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/getnora-io%2Fnora/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/getnora-io%2Fnora/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/getnora-io%2Fnora/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/getnora-io","download_url":"https://codeload.github.com/getnora-io/nora/tar.gz/refs/heads/main","sbom_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/getnora-io%2Fnora/sbom","scorecard":{"id":1245304,"data":{"date":"2026-03-27T19:12:00Z","repo":{"name":"github.com/getnora-io/nora","commit":"9bf6615a10323c41b4f85b0e43c85f03326c0ce4"},"scorecard":{"version":"v5.3.0","commit":"c22063e786c11f9dd714d777a687ff7c4599b600"},"score":7.3,"checks":[{"name":"Dependency-Update-Tool","score":10,"reason":"update tool detected","details":["Info: detected update tool: Dependabot: .github/dependabot.yml:1"],"documentation":{"short":"Determines if the project uses a dependency update tool.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#dependency-update-tool"}},{"name":"Code-Review","score":0,"reason":"Found 0/30 approved changesets -- score normalized to 0","details":null,"documentation":{"short":"Determines if the project requires human code review before pull requests (aka merge requests) are merged.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#code-review"}},{"name":"Binary-Artifacts","score":10,"reason":"no binaries found in the repo","details":null,"documentation":{"short":"Determines if the project has generated executable (binary) artifacts in the source repository.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#binary-artifacts"}},{"name":"Security-Policy","score":10,"reason":"security policy file detected","details":["Info: security policy file detected: SECURITY.md:1","Info: Found linked content: SECURITY.md:1","Info: Found disclosure, vulnerability, and/or timelines in security policy: SECURITY.md:1","Info: Found text in security policy: SECURITY.md:1"],"documentation":{"short":"Determines if the project has published a security policy.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#security-policy"}},{"name":"Dangerous-Workflow","score":10,"reason":"no dangerous workflow patterns detected","details":null,"documentation":{"short":"Determines if the project's GitHub Action workflows avoid dangerous patterns.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#dangerous-workflow"}},{"name":"Maintained","score":0,"reason":"project was created within the last 90 days. Please review its contents carefully","details":["Warn: Repository was created within the last 90 days."],"documentation":{"short":"Determines if the project is \"actively maintained\".","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#maintained"}},{"name":"Token-Permissions","score":10,"reason":"GitHub workflow tokens follow principle of least privilege","details":["Info: jobLevel 'contents' permission set to 'read': .github/workflows/ci.yml:39","Info: jobLevel 'contents' permission set to 'read': .github/workflows/ci.yml:75","Info: jobLevel 'contents' permission set to 'read': .github/workflows/codeql.yml:18","Info: jobLevel 'contents' permission set to 'read': .github/workflows/release.yml:161","Info: jobLevel 'packages' permission set to 'read': .github/workflows/release.yml:162","Warn: jobLevel 'contents' permission set to 'write': .github/workflows/release.yml:203","Info: jobLevel 'contents' permission set to 'read': .github/workflows/release.yml:19","Info: jobLevel 'contents' permission set to 'read': .github/workflows/scorecard.yml:18","Info: jobLevel 'actions' permission set to 'read': .github/workflows/scorecard.yml:19","Info: topLevel permissions set to 'read-all': .github/workflows/ci.yml:9","Info: topLevel permissions set to 'read-all': .github/workflows/codeql.yml:11","Info: topLevel permissions set to 'read-all': .github/workflows/release.yml:7","Info: topLevel permissions set to 'read-all': .github/workflows/scorecard.yml:9"],"documentation":{"short":"Determines if the project's workflows follow the principle of least privilege.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#token-permissions"}},{"name":"Pinned-Dependencies","score":9,"reason":"dependency not pinned by hash detected -- score normalized to 9","details":["Warn: third-party GitHubAction not pinned by hash: .github/workflows/scorecard.yml:27: update your workflow using https://app.stepsecurity.io/secureworkflow/getnora-io/nora/scorecard.yml/main?enable=pin","Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/scorecard.yml:35: update your workflow using https://app.stepsecurity.io/secureworkflow/getnora-io/nora/scorecard.yml/main?enable=pin","Warn: npmCommand not pinned by hash: tests/smoke.sh:265","Warn: pipCommand not pinned by hash: .github/workflows/ci.yml:108","Info:  16 out of  17 GitHub-owned GitHubAction dependencies pinned","Info:  26 out of  27 third-party GitHubAction dependencies pinned","Info:   4 out of   4 containerImage dependencies pinned","Info:   0 out of   1 npmCommand dependencies pinned","Info:   0 out of   1 pipCommand dependencies pinned"],"documentation":{"short":"Determines if the project has declared and pinned the dependencies of its build process.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#pinned-dependencies"}},{"name":"CII-Best-Practices","score":5,"reason":"badge detected: Passing","details":null,"documentation":{"short":"Determines if the project has an OpenSSF (formerly CII) Best Practices Badge.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#cii-best-practices"}},{"name":"Signed-Releases","score":4,"reason":"3 out of the last 5 releases have a total of 3 signed artifacts.","details":["Info: signed release artifact: nora-linux-amd64.sig: https://github.com/getnora-io/nora/releases/tag/v0.2.35","Info: signed release artifact: nora-linux-amd64.sig: https://github.com/getnora-io/nora/releases/tag/v0.2.34","Info: signed release artifact: nora-linux-amd64.sig: https://github.com/getnora-io/nora/releases/tag/v0.2.33","Warn: release artifact v0.2.32 not signed: https://api.github.com/repos/getnora-io/nora/releases/298358927","Warn: release artifact v0.2.31 not signed: https://api.github.com/repos/getnora-io/nora/releases/297452384","Warn: release artifact v0.2.35 does not have provenance: https://api.github.com/repos/getnora-io/nora/releases/299628595","Warn: release artifact v0.2.34 does not have provenance: https://api.github.com/repos/getnora-io/nora/releases/299578817","Warn: release artifact v0.2.33 does not have provenance: https://api.github.com/repos/getnora-io/nora/releases/298920031","Warn: release artifact v0.2.32 does not have provenance: https://api.github.com/repos/getnora-io/nora/releases/298358927","Warn: release artifact v0.2.31 does not have provenance: https://api.github.com/repos/getnora-io/nora/releases/297452384"],"documentation":{"short":"Determines if the project cryptographically signs release artifacts.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#signed-releases"}},{"name":"Vulnerabilities","score":10,"reason":"0 existing vulnerabilities detected","details":null,"documentation":{"short":"Determines if the project has open, known unfixed vulnerabilities.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#vulnerabilities"}},{"name":"SAST","score":10,"reason":"SAST tool is run on all commits","details":["Info: SAST configuration detected: CodeQL","Info: all commits (2) are checked with a SAST tool"],"documentation":{"short":"Determines if the project uses static code analysis.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#sast"}},{"name":"Packaging","score":10,"reason":"packaging workflow detected","details":["Info: Project packages its releases by way of GitHub Actions.: .github/workflows/ci.yml:153"],"documentation":{"short":"Determines if the project is published as a package that others can easily download, install, easily update, and uninstall.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#packaging"}},{"name":"Branch-Protection","score":3,"reason":"branch protection is not maximal on development and all release branches","details":["Info: 'allow deletion' disabled on branch 'main'","Info: 'force pushes' disabled on branch 'main'","Warn: 'branch protection settings apply to administrators' is disabled on branch 'main'","Warn: 'stale review dismissal' is disabled on branch 'main'","Warn: branch 'main' does not require approvers","Warn: codeowners review is not required on branch 'main'","Warn: 'last push approval' is disabled on branch 'main'","Warn: 'up-to-date branches' is disabled on branch 'main'","Info: status check found to merge onto on branch 'main'","Info: PRs are required in order to make changes on branch 'main'"],"documentation":{"short":"Determines if the default and release branches are protected with GitHub's branch protection settings.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#branch-protection"}},{"name":"License","score":10,"reason":"license file detected","details":["Info: project has a license file: LICENSE:0","Info: FSF or OSI recognized license: MIT License: LICENSE:0"],"documentation":{"short":"Determines if the project has defined a license.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#license"}},{"name":"Fuzzing","score":10,"reason":"project is fuzzed","details":["Info: ClusterFuzzLite integration found","Info: RustCargoFuzzer integration found: fuzz/fuzz_targets/fuzz_docker_manifest.rs:2","Info: RustCargoFuzzer integration found: fuzz/fuzz_targets/fuzz_validation.rs:2"],"documentation":{"short":"Determines if the project uses fuzzing.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#fuzzing"}},{"name":"CI-Tests","score":10,"reason":"2 out of 2 merged PRs checked by a CI test -- score normalized to 10","details":null,"documentation":{"short":"Determines if the project runs tests before pull requests are merged.","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#ci-tests"}},{"name":"Contributors","score":3,"reason":"project has 1 contributing companies or organizations -- score normalized to 3","details":["Info: found contributions from: devitway"],"documentation":{"short":"Determines if the project has a set of contributors from multiple organizations (e.g., companies).","url":"https://github.com/ossf/scorecard/blob/c22063e786c11f9dd714d777a687ff7c4599b600/docs/checks.md#contributors"}}]},"last_synced_at":"2026-03-27T19:37:33.643Z","repository_id":334579546,"created_at":"2026-03-27T19:37:33.643Z","updated_at":"2026-03-27T19:37:33.643Z"},"host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":286080680,"owners_count":32361477,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2026-04-27T20:07:02.737Z","status":"ssl_error","status_checked_at":"2026-04-27T20:07:00.910Z","response_time":128,"last_error":"SSL_connect returned=1 errno=0 peeraddr=140.82.121.5:443 state=error: unexpected eof while reading","robots_txt_status":"success","robots_txt_updated_at":"2025-07-24T06:49:26.215Z","robots_txt_url":"https://github.com/robots.txt","online":false,"can_crawl_api":true,"host_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub","repositories_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories","repository_names_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repository_names","owners_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners"}},"keywords":["air-gapped","artifact-registry","artifactory-alternative","cargo-registry","conan","curation","docker-registry","maven-repository","nexus-alternative","npm-registry","nuget","oci-registry","pypi-registry","registry-proxy","rubygems","rust","self-hosted","single-binary","supply-chain-security","terraform-registry"],"created_at":"2026-01-26T13:07:11.267Z","updated_at":"2026-04-28T01:02:23.545Z","avatar_url":"https://github.com/getnora-io.png","language":"Rust","readme":"# NORA\n\n**The artifact registry that grows with you.** Starts with `docker run`, scales to enterprise.\n\n```bash\ndocker run -d -p 4000:4000 -v nora-data:/data ghcr.io/getnora-io/nora:latest\n```\n\nOpen [http://localhost:4000/ui/](http://localhost:4000/ui/) — your registry is ready.\n\n\u003cp align=\"center\"\u003e\n  \u003cimg src=\".github/assets/dashboard.png\" alt=\"NORA Dashboard\" width=\"960\" /\u003e\n\u003c/p\u003e\n\n## Why NORA\n\n- **Zero-config** — single binary, no database, no dependencies. `docker run` and it works.\n- **13 registries** — Docker, Maven, npm, PyPI, Cargo, Go, Raw, RubyGems, Terraform, Ansible Galaxy, NuGet, Pub (Dart/Flutter), Conan (C/C++).\n- **Secure by default** — [OpenSSF Scorecard](https://scorecard.dev/viewer/?uri=github.com/getnora-io/nora), signed releases, SBOM, fuzz testing, 850 tests.\n\n[![Release](https://img.shields.io/github/v/release/getnora-io/nora)](https://github.com/getnora-io/nora/releases)\n[![License: MIT](https://img.shields.io/badge/License-MIT-blue.svg)](LICENSE)\n[![Artifact Hub](https://img.shields.io/endpoint?url=https://artifacthub.io/badge/repository/nora)](https://artifacthub.io/packages/helm/nora/nora)\n\n**\u003c 25 MB** binary | **\u003c 100 MB** RAM | **3s** startup | **13** registries\n\n## Supported Registries\n\n| Registry | Mount Point | Upstream Proxy | Auth |\n|----------|------------|----------------|------|\n| Docker Registry v2 | `/v2/` | Docker Hub, GHCR, any OCI, Helm OCI | ✓ |\n| Maven | `/maven2/` | Maven Central, custom | ✓ |\n| npm | `/npm/` | npmjs.org, custom | ✓ |\n| Cargo | `/cargo/` | crates.io | ✓ |\n| PyPI | `/simple/` | pypi.org, custom | ✓ |\n| Go Modules | `/go/` | proxy.golang.org, custom | ✓ |\n| Raw files | `/raw/` | — | ✓ |\n| RubyGems | `/gems/` | rubygems.org | ✓ |\n| Terraform | `/terraform/` | registry.terraform.io | ✓ |\n| Ansible Galaxy | `/ansible/` | galaxy.ansible.com | ✓ |\n| NuGet | `/nuget/` | api.nuget.org | ✓ |\n| Pub (Dart/Flutter) | `/pub/` | pub.dev | ✓ |\n| Conan (C/C++) | `/conan/` | ConanCenter | ✓ |\n\n\u003e **Helm charts** work via the Docker/OCI endpoint — `helm push`/`pull` with `--plain-http` or behind TLS reverse proxy.\n\n## Quick Start\n\n### Docker (Recommended)\n\n```bash\ndocker run -d -p 4000:4000 -v nora-data:/data ghcr.io/getnora-io/nora:latest\n```\n\n### Binary\n\n```bash\ncurl -fsSL https://github.com/getnora-io/nora/releases/latest/download/nora-linux-amd64 -o nora\nchmod +x nora \u0026\u0026 ./nora\n```\n\n### Kubernetes (Helm)\n\n```bash\nhelm repo add nora https://getnora-io.github.io/helm-charts\nhelm install nora nora/nora\n```\n\n### From Source\n\n```bash\ncargo install nora-registry\nnora\n```\n\n## Usage\n\n```bash\n# Docker\ndocker tag myapp:latest localhost:4000/myapp:latest\ndocker push localhost:4000/myapp:latest\n\n# npm\nnpm config set registry http://localhost:4000/npm/\nnpm publish\n\n# Go\nGOPROXY=http://localhost:4000/go go get golang.org/x/text@latest\n```\n\nSee [full documentation](https://getnora.dev) for all registries.\n\n## Features\n\n- **Web UI** — dashboard with search, browse, i18n (EN/RU)\n- **Proxy \u0026 Cache** — transparent proxy to upstream registries with local cache\n- **Curation** — blocklist, allowlist, namespace isolation, integrity verification, min-release-age filter\n- **Token RBAC** — read/write/admin roles, expiry tracking, deferred last_used flush\n- **Mirror CLI** — offline sync for air-gapped environments (`nora mirror`)\n- **Backup \u0026 Restore** — `nora backup` / `nora restore`\n- **S3 Storage** — MinIO, AWS S3, any S3-compatible backend\n- **Prometheus Metrics** — `/metrics` endpoint\n- **Rate Limiting** — configurable per-endpoint rate limits\n\n## Configuration\n\nNORA works out of the box. For advanced setup — auth, S3, retention, curation — see [getnora.dev/configuration](https://getnora.dev/configuration/settings/).\n\n```bash\n# Auth\ndocker run -d -p 4000:4000 \\\n  -v nora-data:/data \\\n  -v ./users.htpasswd:/data/users.htpasswd \\\n  -e NORA_AUTH_ENABLED=true \\\n  ghcr.io/getnora-io/nora:latest\n```\n\n```bash\n# Curation — block packages younger than 7 days\ndocker run -d -p 4000:4000 \\\n  -v nora-data:/data \\\n  -e NORA_CURATION_MODE=enforce \\\n  -e NORA_CURATION_MIN_RELEASE_AGE=7d \\\n  -e NORA_CURATION_ALLOWLIST_PATH=/data/allowlist.json \\\n  ghcr.io/getnora-io/nora:latest\n```\n\n## Performance\n\n| Metric | NORA | Nexus | JFrog |\n|--------|------|-------|-------|\n| Startup | \u003c 3s | 30-60s | 30-60s |\n| Memory | \u003c 100 MB | 2-4 GB | 2-4 GB |\n| Binary | \u003c 25 MB | 600+ MB | 1+ GB |\n\n## Roadmap\n\n- ~~Mirror CLI~~ ✅ v0.4.0\n- ~~Garbage Collection \u0026 Retention~~ ✅ v0.6.0\n- ~~Helm Chart~~ ✅ v0.6.1\n- ~~Signed releases \u0026 SBOM~~ ✅ v0.6.4\n- ~~Curation layer~~ ✅ v0.7.0\n- ~~13 registry formats~~ ✅ v0.7.0\n- ~~Min Release Age~~ ✅ v0.7.1\n- **OIDC / Workload Identity** — zero-secret auth for GitHub Actions, GitLab CI\n- **Image Signing Policy** — cosign verification on upstream pulls\n\nSee [CHANGELOG.md](CHANGELOG.md) for release history.\n\n## Security \u0026 Trust\n\n[![OpenSSF Scorecard](https://api.scorecard.dev/projects/github.com/getnora-io/nora/badge)](https://scorecard.dev/viewer/?uri=github.com/getnora-io/nora)\n[![CII Best Practices](https://www.bestpractices.dev/projects/12207/badge)](https://www.bestpractices.dev/projects/12207)\n[![Coverage](https://img.shields.io/endpoint?url=https://gist.githubusercontent.com/devitway/0f0538f1ed16d5d9951e4f2d3f79b699/raw/nora-coverage.json)](https://github.com/getnora-io/nora/actions/workflows/ci.yml)\n[![CI](https://img.shields.io/github/actions/workflow/status/getnora-io/nora/ci.yml?label=CI)](https://github.com/getnora-io/nora/actions)\n\nSee [SECURITY.md](SECURITY.md) for vulnerability reporting.\n\n## Documentation\n\nFull documentation: **https://getnora.dev**\n\n## Author\n\nCreated and maintained by [Pavel Volkov](https://github.com/devitway)\n\n[![Docs](https://img.shields.io/badge/docs-getnora.dev-green?logo=gitbook)](https://getnora.dev)\n[![Telegram](https://img.shields.io/badge/Telegram-Community-blue?logo=telegram)](https://t.me/getnora)\n[![GitHub Stars](https://img.shields.io/github/stars/getnora-io/nora?style=flat\u0026logo=github)](https://github.com/getnora-io/nora/stargazers)\n\n## Contributing\n\nNORA welcomes contributions! See [CONTRIBUTING.md](CONTRIBUTING.md) for guidelines.\n\n## License\n\nMIT License — see [LICENSE](LICENSE)\n\nCopyright (c) 2026 The NORA Authors\n","funding_links":[],"categories":["Registries","Docker Images"],"sub_categories":["Web programming","Registry"],"project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fgetnora-io%2Fnora","html_url":"https://awesome.ecosyste.ms/projects/github.com%2Fgetnora-io%2Fnora","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Fgetnora-io%2Fnora/lists"}