Skip to content

Commit

Permalink
fix: perform device flow from CLI
Browse files Browse the repository at this point in the history
  • Loading branch information
nsklikas committed May 22, 2024
1 parent ec1e61f commit 7a29a5a
Show file tree
Hide file tree
Showing 2 changed files with 109 additions and 0 deletions.
108 changes: 108 additions & 0 deletions cmd/cmd_perform_device_flow.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,108 @@
// Copyright © 2022 Ory Corp
// SPDX-License-Identifier: Apache-2.0

package cmd

import (
"context"
"fmt"
"os"
"strings"

"github.com/ory/hydra/v2/cmd/cliclient"

"github.com/spf13/cobra"
"golang.org/x/oauth2"

"github.com/ory/x/cmdx"
"github.com/ory/x/flagx"
"github.com/ory/x/urlx"
)

func NewPerformDeviceCodeCmd() *cobra.Command {
cmd := &cobra.Command{
Use: "device-code",
Example: "{{ .CommandPath }} --client-id ... --client-secret ...",
Short: "An exemplary OAuth 2.0 Client performing the OAuth 2.0 Device Code Flow",
Long: `Performs the device code flow. Useful for getting an access token and an ID token in machines without a browser.
The client that will be used MUST support the "client_secret_post" token-endpoint-auth-method
`,
RunE: func(cmd *cobra.Command, args []string) error {
client, endpoint, err := cliclient.NewClient(cmd)
if err != nil {
return err
}

endpoint = cliclient.GetOAuth2URLOverride(cmd, endpoint)

ctx := context.WithValue(cmd.Context(), oauth2.HTTPClient, client)
scopes := flagx.MustGetStringSlice(cmd, "scope")
deviceAuthUrl := flagx.MustGetString(cmd, "device-auth-url")
tokenUrl := flagx.MustGetString(cmd, "token-url")
audience := flagx.MustGetStringSlice(cmd, "audience")

clientID := flagx.MustGetString(cmd, "client-id")
if clientID == "" {
_, _ = fmt.Fprint(cmd.OutOrStdout(), cmd.UsageString())
_, _ = fmt.Fprintln(cmd.OutOrStdout(), "Please provide a Client ID using --client-id flag, or OAUTH2_CLIENT_ID environment variable.")
return cmdx.FailSilently(cmd)
}

clientSecret := flagx.MustGetString(cmd, "client-secret")

if deviceAuthUrl == "" {
deviceAuthUrl = urlx.AppendPaths(endpoint, "/oauth2/device/auth").String()
}

if tokenUrl == "" {
tokenUrl = urlx.AppendPaths(endpoint, "/oauth2/token").String()
}

conf := oauth2.Config{
ClientID: clientID,
ClientSecret: clientSecret,
Endpoint: oauth2.Endpoint{
DeviceAuthURL: deviceAuthUrl,
TokenURL: tokenUrl,
},
Scopes: scopes,
}

deviceAuthResponse, err := conf.DeviceAuth(
ctx,
oauth2.SetAuthURLParam("audience", strings.Join(audience, "+")),
oauth2.SetAuthURLParam("client_secret", clientSecret),
)
if err != nil {
cmdx.Fatalf("Failed to perform the device authorization request", err.Error())
}

fmt.Fprintf(
cmd.OutOrStdout(),
"To login please go to %s",
deviceAuthResponse.VerificationURIComplete,
)

token, err := conf.DeviceAccessToken(ctx, deviceAuthResponse)
if err != nil {
cmdx.Fatalf("Failed to perform the device token request: %e", err.Error())
}

fmt.Println("Successfully signed in!")

cmdx.PrintRow(cmd, outputOAuth2Token(*token))
return nil
},
}

cmd.Flags().StringSlice("scope", []string{"offline", "openid"}, "Request OAuth2 scope")

cmd.Flags().String("client-id", os.Getenv("OAUTH2_CLIENT_ID"), "Use the provided OAuth 2.0 Client ID, defaults to environment variable OAUTH2_CLIENT_ID")
cmd.Flags().String("client-secret", os.Getenv("OAUTH2_CLIENT_SECRET"), "Use the provided OAuth 2.0 Client Secret, defaults to environment variable OAUTH2_CLIENT_SECRET")

cmd.Flags().StringSlice("audience", []string{}, "Request a specific OAuth 2.0 Access Token Audience")
cmd.Flags().String("device-auth-url", "", "Usually it is enough to specify the `endpoint` flag, but if you want to force the device authorization url, use this flag")
cmd.Flags().String("token-url", "", "Usually it is enough to specify the `endpoint` flag, but if you want to force the token url, use this flag")

return cmd
}
1 change: 1 addition & 0 deletions cmd/root.go
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,7 @@ func RegisterCommandRecursive(parent *cobra.Command, slOpts []servicelocatorx.Op
performCmd.AddCommand(
NewPerformClientCredentialsCmd(),
NewPerformAuthorizationCodeCmd(),
NewPerformDeviceCodeCmd(),
)

revokeCmd := NewRevokeCmd()
Expand Down

0 comments on commit 7a29a5a

Please sign in to comment.