Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

feat(cli): Add import sub commmand for project. #594

Merged

Conversation

muntaxir4
Copy link
Contributor

@muntaxir4 muntaxir4 commented Dec 22, 2024

User description

Description

@keyshade/secret-scan: Create a function to detect secrets and variables from JS object containing keys and values.

cli: Add feature to import secrets and variables from .env file into Keyshade.

Fixes #502
Fixes #503

Developer's checklist

  • My PR follows the style guidelines of this project
  • I have performed a self-check on my work

If changes are made in the code:

  • I have followed the coding guidelines
  • My changes in code generate no new warnings
  • My changes are breaking another fix/feature of the project
  • I have added test cases to show that my feature works
  • I have added relevant screenshots in my PR
  • There are no UI/UX issues

Documentation Update

  • This PR requires an update to the documentation at docs.keyshade.xyz
  • I have made the necessary updates to the documentation, or no documentation changes are required.

PR Type

Enhancement


Description

  • Added new CLI command keyshade project import to import secrets and variables from .env files
  • Implemented secret detection functionality in @keyshade/secret-scan package:
    • New detectObject function to analyze key-value pairs
    • Classifies values as secrets or variables using detection rules
    • Added comprehensive test coverage
  • Added interactive workflow:
    • Displays detected secrets and variables
    • Requires user confirmation
    • Allows environment selection
  • Handles errors gracefully during secret/variable creation

Changes walkthrough 📝

Relevant files
Enhancement
import.project.ts
Add import command for environment secrets and variables 

apps/cli/src/commands/project/import.project.ts

  • Added new command to import secrets and variables from .env file
  • Implemented file reading, parsing, and detection of secrets/variables
  • Added interactive confirmation and environment selection
  • Handles creation of secrets and variables with error handling
  • +144/-0 
    index.ts
    Add function to detect secrets from objects                           

    packages/secret-scan/src/index.ts

  • Added detectObject method to scan key-value pairs
  • Classifies values as secrets or variables based on detection rules
  • +18/-1   
    project.command.ts
    Register import command in project commands                           

    apps/cli/src/commands/project.command.ts

    • Added ImportEnvs command to project commands list
    +3/-1     
    index.d.ts
    Add types for secret detection results                                     

    packages/secret-scan/src/types/index.d.ts

    • Added ScanObjectResult interface for secret detection results
    +5/-0     
    Tests
    detect-object.test.ts
    Add tests for secret detection from objects                           

    packages/secret-scan/src/test/detect-object.test.ts

  • Added test cases for detecting secrets and variables from objects
  • Tests empty input, only secrets, only variables scenarios
  • Validates correct classification of different types of values
  • +63/-0   
    Dependencies
    package.json
    Add dotenv package dependency                                                       

    apps/cli/package.json

  • Added dotenv package dependency
  • Reordered dependencies alphabetically
  • +3/-2     

    💡 PR-Agent usage: Comment /help "your question" on any pull request to receive relevant information

    Copy link
    Contributor

    PR Reviewer Guide 🔍

    Here are some key observations to aid the review process:

    🎫 Ticket compliance analysis 🔶

    503 - Partially compliant

    Compliant requirements:

    • Import command added with .env file input support
    • Uses secret-scan package to detect secrets/variables
    • Uploads to specified project
    • Shows confirmation prompt
    • Prints success/error messages

    Non-compliant requirements:

    • Error handling stops entire operation on failure instead of continuing

    502 - Fully compliant

    Compliant requirements:

    • detectObject function created to scan objects
    • Detects and flags secrets vs variables
    • Returns proper JSON structure with arrays
    ⏱️ Estimated effort to review: 3 🔵🔵🔵⚪⚪
    🧪 PR contains tests
    🔒 No security concerns identified
    ⚡ Recommended focus areas for review

    Error Handling

    The import process stops on first error instead of continuing and collecting errors in an array as specified in requirements

              Logger.error(
                `Failed to create secret for ${key} : ${error.message}.\nAborting!`
              )
              throw Error('Error while creating secret')
            }
    Missing Validation

    No validation of project slug and environment slug existence before attempting to create secrets/variables

          const environmentSlug = (await text({
            message: 'Enter the environment slug to import to:'
          })) as string
    
          Logger.info(
            `Importing secrets and variables to project: ${projectSlug} and environment: ${environmentSlug} with default settings`
          )

    Copy link
    Contributor

    codiumai-pr-agent-free bot commented Dec 22, 2024

    PR Code Suggestions ✨

    Explore these optional code suggestions:

    CategorySuggestion                                                                                                                                    Score
    Possible issue
    ✅ Validate parsed environment variables to ensure non-empty input before processing

    Add validation for empty or invalid environment variables before processing. Check
    if envVariables object contains any entries after parsing.

    apps/cli/src/commands/project/import.project.ts [49-51]

     const envVariables = dotenv.parse(envFileContent)
    +if (Object.keys(envVariables).length === 0) {
    +  throw new Error('No environment variables found in the provided file')
    +}
     const secretsAndVariables = secretDetector.detectObject(envVariables)

    [Suggestion has been applied]

    Suggestion importance[1-10]: 8

    Why: This is a critical validation that prevents processing empty files and provides clear error feedback. Missing this check could lead to silent failures or unnecessary processing.

    8
    ✅ Verify file existence before attempting to read its contents

    Add file existence check before attempting to read the .env file to provide a
    clearer error message.

    apps/cli/src/commands/project/import.project.ts [44-47]

    -const envFileContent = await fs.readFile(
    -  path.resolve(dotEnvPath),
    -  'utf-8'
    -)
    +const resolvedPath = path.resolve(dotEnvPath)
    +const exists = await fs.access(resolvedPath).then(() => true).catch(() => false)
    +if (!exists) {
    +  throw new Error(`The .env file does not exist at path: ${resolvedPath}`)
    +}
    +const envFileContent = await fs.readFile(resolvedPath, 'utf-8')

    [Suggestion has been applied]

    Suggestion importance[1-10]: 7

    Why: Important defensive programming practice that provides better error handling and user feedback when the .env file is missing, preventing cryptic filesystem errors.

    7
    Validate user input to prevent empty or invalid values

    Add validation for the environment slug to ensure it's not empty or just whitespace.

    apps/cli/src/commands/project/import.project.ts [77-79]

     const environmentSlug = (await text({
       message: 'Enter the environment slug to import to:'
     })) as string
    +if (!environmentSlug?.trim()) {
    +  throw new Error('Environment slug cannot be empty')
    +}
    • Apply this suggestion
    Suggestion importance[1-10]: 7

    Why: Essential input validation that prevents potential issues with empty environment slugs, which could cause problems in downstream operations or database entries.

    7

    @muntaxir4
    Copy link
    Contributor Author

    These are the initial changes, and some updates might be required.

    apps/cli/src/commands/project.command.ts Outdated Show resolved Hide resolved
    apps/cli/src/commands/project/import.project.ts Outdated Show resolved Hide resolved
    packages/secret-scan/src/types/index.d.ts Outdated Show resolved Hide resolved
    @muntaxir4 muntaxir4 requested a review from rajdip-b December 23, 2024 20:26
    Copy link
    Member

    @rajdip-b rajdip-b left a comment

    Choose a reason for hiding this comment

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

    I tried importing the env file. seems like anything that gets enclosed in "" gets detected as a variable. We need to strip values off "" before scanning

    apps/cli/src/commands/project/import.project.ts Outdated Show resolved Hide resolved
    Copy link

    codecov bot commented Dec 31, 2024

    Codecov Report

    All modified and coverable lines are covered by tests ✅

    Project coverage is 99.76%. Comparing base (ce50743) to head (f62b41c).
    Report is 258 commits behind head on develop.

    Additional details and impacted files
    @@             Coverage Diff             @@
    ##           develop     #594      +/-   ##
    ===========================================
    + Coverage    91.71%   99.76%   +8.05%     
    ===========================================
      Files          111       71      -40     
      Lines         2510      432    -2078     
      Branches       469        3     -466     
    ===========================================
    - Hits          2302      431    -1871     
    + Misses         208        1     -207     
    Flag Coverage Δ
    api-e2e-tests ?
    secret-scan 99.76% <100.00%> (?)

    Flags with carried forward coverage won't be shown. Click here to find out more.

    ☔ View full report in Codecov by Sentry.
    📢 Have feedback on the report? Share it here.

    @muntaxir4 muntaxir4 requested a review from rajdip-b January 1, 2025 21:46
    packages/secret-scan/src/types/index.d.ts Outdated Show resolved Hide resolved
    packages/secret-scan/src/index.ts Outdated Show resolved Hide resolved
    apps/cli/src/commands/project/import.project.ts Outdated Show resolved Hide resolved
    apps/cli/src/commands/project/import.project.ts Outdated Show resolved Hide resolved
    @muntaxir4
    Copy link
    Contributor Author

    Also added some ignored Extensions like media files to scan command. As it was scanning images as well.

    @muntaxir4 muntaxir4 requested a review from rajdip-b January 4, 2025 21:20
    @rajdip-b rajdip-b merged commit 9896f27 into keyshade-xyz:develop Jan 7, 2025
    5 checks passed
    @muntaxir4 muntaxir4 deleted the feat/import-envs-to-project branch January 19, 2025 20:39
    rajdip-b pushed a commit that referenced this pull request Jan 23, 2025
    ## [2.9.0](v2.8.0...v2.9.0) (2025-01-23)
    
    ### 🚀 Features
    
    * **api-client:** Get all workspace invitation ([#619](#619)) ([8a23850](8a23850))
    * **api,cli,api-client,schema:** Enhance permissions to allow filtering by environments through project roles ([#599](#599)) ([030b539](030b539))
    * **api:** Add `ADMIN` authority for API keys ([#609](#609)) ([fb6aba7](fb6aba7))
    * **api:** Add email template for inviting user to workspace ([#480](#480)) ([f5ddf7a](f5ddf7a))
    * **api:** Add email template for sending OTP to the user ([#582](#582)) ([cb6bbcb](cb6bbcb))
    * **api:** Add endpoint to fetch all workspace invitations for a user ([#586](#586)) ([d45417a](d45417a))
    * **api:** Add logout endpoint to clear token cookie ([#581](#581)) ([27f81ba](27f81ba))
    * **api:** Add slack integration ([#531](#531)) ([fe124d8](fe124d8))
    * **cli:** Add CLI command to check version flag using `--version` or `-v` ([#650](#650)) ([31b5efe](31b5efe))
    * **cli:** Add functionality to operate on Workspace Membership ([#589](#589)) ([0fde62b](0fde62b))
    * **cli:** Add import sub commmand for project. ([#594](#594)) ([9896f27](9896f27))
    * **cli:** Add List-invitation command ([#633](#633)) ([874f8c2](874f8c2))
    * **cli:** Added keyshade command to cli ([cf260ae](cf260ae))
    * **cli:** Api health probe ([#645](#645)) ([dd854f4](dd854f4))
    * **cli:** Create basic README.md ([a1b74e9](a1b74e9))
    * **cli:** Log publicKey, privacyKey, & accessLevel after project creation ([#623](#623)) ([5d5b329](5d5b329))
    * **cli:** Supports to specify environment(s) and its optional description ([#634](#634)) ([62083b1](62083b1))
    * **cli:** update README with feature and installation ([#644](#644)) ([a4d2a6a](a4d2a6a))
    * **platform:** Add a new [secure] and added loader on project screen ([#603](#603)) ([c3a08cc](c3a08cc))
    * **platform:** Add new variables to a project ([#593](#593)) ([d6c6252](d6c6252))
    * **platform:** Delete variable from a project ([#600](#600)) ([e64a738](e64a738))
    * **platform:** Edit existing variables in a project ([#602](#602)) ([bb48f6c](bb48f6c))
    * **platform:** Show all the existing variables inside a project ([#591](#591)) ([5276bb8](5276bb8))
    * **platofrm:** Added online/offline status detection in the platform ([#585](#585)) ([89aa84f](89aa84f))
    * **schema:** Add workspace invitation schema ([#612](#612)) ([1a5721b](1a5721b))
    * **web:** Add Google Analytics integration ([#649](#649)) ([397d6da](397d6da))
    
    ### 🐛 Bug Fixes
    
    * **api:** Empty name `""` accepted as a valid name while creating environments ([#583](#583)) ([a33f4b5](a33f4b5))
    * **api:** Enable global project access ([#580](#580)) ([b3a0309](b3a0309))
    * **api:** Update build command ([0ddfa59](0ddfa59))
    * **cli:** Add keywords for improved package discoverability ([#641](#641)) ([57ce10b](57ce10b))
    * **cli:** Check for .keyshade dir if profile isn't found ([#636](#636)) ([a69665d](a69665d))
    * **cli:** Create project --store-private-key option default value removed ([#638](#638)) ([20f16c6](20f16c6))
    * **cli:** Fixed binary path in package.json ([e531af0](e531af0))
    * **cli:** Fixed binary path in package.json ([81d674d](81d674d))
    * **cli:** Incorrect message on listing projects ([#624](#624)) ([eeffa42](eeffa42))
    * **cli:** Module errors ([d3432c5](d3432c5))
    * **cli:** Module errors ([a639065](a639065))
    * **cli:** Module errors ([a7742b1](a7742b1))
    * **cli:** Module errors ([e96300e](e96300e))
    * **cli:** Profile name now can use - and _ and updated error message ([#639](#639)) ([00dd66a](00dd66a))
    * **cli:** Prompt users for all values if no option set and show default values ([#640](#640)) ([fe862ab](fe862ab))
    * **docker:** Update build script ([40ef3e2](40ef3e2))
    * **platform:** Check if `Env. Name` is left empty ([#646](#646)) ([5f3fac8](5f3fac8))
    * **platform:** Clickable Workspaces combobox options ([#630](#630)) ([acc96f7](acc96f7))
    * **platform:** Optimized user update request body ([#605](#605)) ([ee1adf0](ee1adf0))
    * **platform:** Type error in navbar ([8199de8](8199de8))
    * **README:** Update Discord badge ([6f9382e](6f9382e))
    * **schema:** Add versions field to project [secure]s and variables response ([#590](#590)) ([755ea46](755ea46))
    
    ### 📚 Documentation
    
    * **cli:** Update changelog to include missed out changes ([8910c5c](8910c5c))
    * Updated alignment of pictures in API Testing page ([5d69223](5d69223))
    * Updated alignment of pictures in API Testing page ([e31eeca](e31eeca))
    
    ### 🔧 Miscellaneous Chores
    
    * Add Sentry and update CI ([#653](#653)) ([ca96862](ca96862))
    * **ci:** Add CLI deployment script ([51de9d1](51de9d1))
    * **ci:** Add internal package dependencies to existing workflows ([#592](#592)) ([a9fc39e](a9fc39e))
    * **ci:** Add scope ([8ef89a8](8ef89a8))
    * **ci:** Bug fix in workflow ([d583a46](d583a46))
    * **ci:** Bug fix in workflow ([eb9d60f](eb9d60f))
    * **ci:** Bump version to 2.2.0 ([951bd14](951bd14))
    * **ci:** Deploy DB migrations ([ea1beed](ea1beed))
    * **ci:** Fixed chaining and scripts ([6a009eb](6a009eb))
    * **ci:** Fixed environment name ([172c348](172c348))
    * **ci:** Fixed errors ([f28e3b5](f28e3b5))
    * **ci:** Minor fixes ([c7f05a0](c7f05a0))
    * **ci:** Push docker images of platform and API to ACR ([5f79dd7](5f79dd7))
    * **ci:** Remove npm ci ([3d45a4c](3d45a4c))
    * **ci:** Remove pnpm cache ([f45970c](f45970c))
    * **ci:** Update app redeployment ([18cf765](18cf765))
    * **ci:** Update web deployment to push to ACR ([26d4bed](26d4bed))
    * **cli:** Bumped CLI to v2.4.0 ([09efcd9](09efcd9))
    * **cli:** Rearranged dependency ([b6e344d](b6e344d))
    * **cli:** Removed changeset ([6c436de](6c436de))
    * **cli:** Update package name ([23552a1](23552a1))
    * **cli:** Update package name ([480cf54](480cf54))
    * **cli:** Update package.json ([871679a](871679a))
    * **cli:** Updated build scripts ([2e2b42d](2e2b42d))
    * **docker:** Update port of platform docker build ([c79d886](c79d886))
    * Housekeeping ([2ed31c0](2ed31c0))
    * **platoform:** Swapped all legacy API calls with `@keyshade/api-client` ([#584](#584)) ([c600db7](c600db7))
    * Removed .postman folder ([4b2b675](4b2b675))
    * Reverted back registry ([1699a89](1699a89))
    * **schema:** Add describe blocks in tests for each kind of schema ([#577](#577)) ([c0763f3](c0763f3))
    * Update npmrc ([9f7f495](9f7f495))
    * Update pipelines; fixed api docker ([3f36a17](3f36a17))
    * Update platform build command ([83a1851](83a1851))
    * **web:** Fix CI ([d1bc740](d1bc740))
    * **web:** Update dockerfile and ci to include google analytics env ([f2df4f4](f2df4f4))
    
    ### 🔨 Code Refactoring
    
    * **cli:** Replace arguments with options ([#615](#615)) ([498f44e](498f44e))
    * **platform:** Optimized codebase ([#629](#629)) ([d411081](d411081))
    * **platform:** Refactor project components ([#626](#626)) ([5b70805](5b70805))
    * **web:** Changed the text in the hero section of the web application ([#579](#579)) ([a92925f](a92925f))
    @rajdip-b
    Copy link
    Member

    🎉 This PR is included in version 2.9.0 🎉

    The release is available on GitHub release

    Your semantic-release bot 📦🚀

    Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
    Projects
    None yet
    2 participants