Railway - 1Password shell plugin

JoqimJoqim
2 min read

Railway shell plugin

Railway's comprehensive infrastructure platform has piqued my interest as it offers the ability to provision infrastructure, develop locally, and seamlessly deploy to the cloud.

Motivated by this, I decided to contribute to the development of the Railway Shell Plugin for 1Password.

This plugin enables users to securely retrieve secrets and credentials stored in their 1Password vault directly within the Railway CLI, simplifying the deployment process and ensuring the highest level of data protection.

Railway uses the railway login command to store the token in a config.json file. It is present at ~/.railway/config.json

The shell plugin returns a CLI executable that mentions the Token.

func RailwayCLI() schema.Executable {
    return schema.Executable{
        Name:      "Railway CLI", // TODO: Check if this is correct
        Runs:      []string{"railway"},
        DocsURL:   sdk.URL("https://railway.com/docs/cli"), // TODO: Replace with actual URL
        NeedsAuth: needsauth.IfAll(
            needsauth.NotForHelpOrVersion(),
            needsauth.NotWithoutArgs(),
            needsauth.NotForExactArgs("config"),
        ),
        Uses: []schema.CredentialUsage{
            {
                Name: credname.Token,
            },
        },
    }
}

The plugin then imports the config file based on the path and then provisions the Token which is decoded from the string obtained.

var defaultEnvVarMapping = map[string]sdk.FieldName{
    "RAILWAY_TOKEN": fieldname.Token,
}

// implement the function below to add support for importing it.
func TryRailwayConfigFile() sdk.Importer {
    return importer.TryFile("~/.railway/config.json", func(ctx context.Context, contents importer.FileContents, in sdk.ImportInput, out *sdk.ImportAttempt) {
        var config struct {
            User struct {
                Token string `json:"token"`
            } `json:"user"`
        }
        if err := contents.ToJSON(&config); err != nil {
            out.AddError(err)
            return
        }

        if config.User.Token == "" {
            return
        }

        out.AddCandidate(sdk.ImportCandidate{
            Fields: map[sdk.FieldName]string{
                fieldname.Token: config.User.Token,
            },
        })
    })
}

By actively participating in the development of Railway's plugin, my objective was to optimize my workflow and enhance the security of my deployments.

Additionally, I acknowledged that my contributions would have a positive impact on the wider Railway community, providing developers with a user-friendly and secure method to handle their secrets within the familiar Railway environment.

#1Password #BuildWith1Password

2
Subscribe to my newsletter

Read articles from Joqim directly inside your inbox. Subscribe to the newsletter, and don't miss out.

Written by

Joqim
Joqim