Skip to content

Commit

Permalink
Adds Gateway API HTTPRoute integration
Browse files Browse the repository at this point in the history
  • Loading branch information
jjaferson committed Jan 31, 2022
1 parent 98434dc commit 74180e6
Show file tree
Hide file tree
Showing 8 changed files with 334 additions and 6 deletions.
17 changes: 17 additions & 0 deletions cmd/generate.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
package cmd

import (
"github.com/spf13/cobra"
)

func generateCommand() *cobra.Command {
cmd := &cobra.Command{
Use: "generate",
Short: "Commands related to kubernetes object generation",
Long: "Commands related to kubernetes object generation",
}

cmd.AddCommand(generateGatewayAPICommand())

return cmd
}
17 changes: 17 additions & 0 deletions cmd/generate_gatewayapi.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
package cmd

import (
"github.com/spf13/cobra"
)

func generateGatewayAPICommand() *cobra.Command {
cmd := &cobra.Command{
Use: "gatewayapi",
Short: "Generate Gataway API resources",
Long: "Generate Gataway API resources",
}

cmd.AddCommand(generateGatewayApiHttpRouteCommand())

return cmd
}
161 changes: 161 additions & 0 deletions cmd/generate_gatewayapi_httproute.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,161 @@
package cmd

import (
"encoding/json"
"fmt"

"github.com/getkin/kin-openapi/openapi3"
"github.com/kuadrant/kuadrantctl/pkg/gatewayapi"
"github.com/kuadrant/kuadrantctl/pkg/utils"
"github.com/spf13/cobra"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
gatewayapiv1alpha2 "sigs.k8s.io/gateway-api/apis/v1alpha2"
)

var (
generateGatewayAPIHTTPRouteOAS string
generateGatewayAPIHTTPRouteHost string
generateGatewayAPIHTTPRouteSvcName string
generateGatewayAPIHTTPRouteSvcNamespace string
generateGatewayAPIHTTPRouteSvcPort int32
generateGatewayAPIHTTPRouteGateways []string
)

//kuadrantctl generate istio virtualservice --namespace myns --oas petstore.yaml --public-host www.kuadrant.io --service-name myservice --gateway kuadrant-gateway
// --namespace myns
// --service-name myservice
// --public-host www.kuadrant.io
// --gateway kuadrant-gateway
// -- service-port 80

func generateGatewayApiHttpRouteCommand() *cobra.Command {
cmd := &cobra.Command{
Use: "httproute",
Short: "Generate Gateway API HTTPRoute from OpenAPI 3.x",
Long: "Generate Gateway API HTTPRoute from OpenAPI 3.x",
RunE: func(cmd *cobra.Command, args []string) error {
return generateGatewayApiHttpRoute(cmd, args)
},
}

// OpenAPI ref
cmd.Flags().StringVar(&generateGatewayAPIHTTPRouteOAS, "oas", "", "/path/to/file.[json|yaml|yml] OR http[s]://domain/resource/path.[json|yaml|yml] OR - (required)")
err := cmd.MarkFlagRequired("oas")
if err != nil {
panic(err)
}

// service ref
cmd.Flags().StringVar(&generateGatewayAPIHTTPRouteSvcName, "service-name", "", "Service name (required)")
err = cmd.MarkFlagRequired("service-name")
if err != nil {
panic(err)
}

// service namespace
cmd.Flags().StringVarP(&generateGatewayAPIHTTPRouteSvcNamespace, "namespace", "n", "", "Service namespace (required)")
err = cmd.MarkFlagRequired("namespace")
if err != nil {
panic(err)
}

// service host
cmd.Flags().StringVar(&generateGatewayAPIHTTPRouteHost, "public-host", "", "Public host (required)")
err = cmd.MarkFlagRequired("public-host")
if err != nil {
panic(err)
}

// service port
cmd.Flags().Int32VarP(&generateGatewayAPIHTTPRouteSvcPort, "port", "p", 80, "Service Port (required)")

// gateway
cmd.Flags().StringSliceVar(&generateGatewayAPIHTTPRouteGateways, "gateway", []string{}, "Gateways (required)")
err = cmd.MarkFlagRequired("gateway")
if err != nil {
panic(err)
}

return cmd
}

func generateGatewayApiHttpRoute(cmd *cobra.Command, args []string) error {
oasDataRaw, err := utils.ReadExternalResource(generateGatewayAPIHTTPRouteOAS)
if err != nil {
return err
}

openapiLoader := openapi3.NewLoader()
doc, err := openapiLoader.LoadFromData(oasDataRaw)
if err != nil {
return err
}

err = doc.Validate(openapiLoader.Context)
if err != nil {
return fmt.Errorf("OpenAPI validation error: %w", err)
}

httpRoute, err := generateGatewayAPIHTTPRoute(cmd, doc)
if err != nil {
return err
}

jsonData, err := json.Marshal(httpRoute)
if err != nil {
return err
}

fmt.Fprintln(cmd.OutOrStdout(), string(jsonData))
return nil
}

func generateGatewayAPIHTTPRoute(cmd *cobra.Command, doc *openapi3.T) (*gatewayapiv1alpha2.HTTPRoute, error) {

//loop through gateway
// https://github.com/getkin/kin-openapi
gatewaysRef := []gatewayapiv1alpha2.ParentRef{}
for _, gateway := range generateGatewayAPIHTTPRouteGateways {
gatewaysRef = append(gatewaysRef, gatewayapiv1alpha2.ParentRef{
Name: gatewayapiv1alpha2.ObjectName(gateway),
})
}

port := gatewayapiv1alpha2.PortNumber(generateGatewayAPIHTTPRouteSvcPort)
service := fmt.Sprintf("%s.%s.svc", generateGatewayAPIHTTPRouteSvcName, generateGatewayAPIHTTPRouteSvcNamespace)
matches, err := gatewayapi.HTTPRouteMatchesFromOAS(doc)
if err != nil {
return nil, err
}

httpRoute := gatewayapiv1alpha2.HTTPRoute{
TypeMeta: v1.TypeMeta{
Kind: "HTTPRoute",
APIVersion: "gateway.networking.k8s.io/v1alpha2",
},
Spec: gatewayapiv1alpha2.HTTPRouteSpec{
CommonRouteSpec: gatewayapiv1alpha2.CommonRouteSpec{
ParentRefs: gatewaysRef,
},
Hostnames: []gatewayapiv1alpha2.Hostname{
gatewayapiv1alpha2.Hostname(generateGatewayAPIHTTPRouteHost),
},
Rules: []gatewayapiv1alpha2.HTTPRouteRule{
{
BackendRefs: []gatewayapiv1alpha2.HTTPBackendRef{
{
BackendRef: gatewayapiv1alpha2.BackendRef{
BackendObjectReference: gatewayapiv1alpha2.BackendObjectReference{
Name: gatewayapiv1alpha2.ObjectName(service),
Port: &port,
},
},
},
},
Matches: matches,
},
},
},
}
return &httpRoute, nil
}
1 change: 1 addition & 0 deletions cmd/root.go
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ func GetRootCmd(args []string) *cobra.Command {
rootCmd.AddCommand(installCommand())
rootCmd.AddCommand(uninstallCommand())
rootCmd.AddCommand(versionCommand())
rootCmd.AddCommand(generateCommand())

loggerOpts := zap.Options{Development: verbose}
logf.SetLogger(zap.New(zap.UseFlagOptions(&loggerOpts)))
Expand Down
36 changes: 36 additions & 0 deletions examples/oas3/gateway-api-petstore.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
---
openapi: "3.0.0"
info:
title: "Pet Store API"
version: "1.0.0"
servers:
- url: https://toplevel.example.io/v1
paths:
/cat:
get:
parameters:
- in: header
name: X-username
schema:
type: string
operationId: "getCat"
responses:
405:
description: "invalid input"
post:
operationId: "postCat"
responses:
405:
description: "invalid input"
/dog:
get:
parameters:
- in: query
name: pathprefix
required: true
schema:
type: string
operationId: "getDog"
responses:
405:
description: "invalid input"
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,6 @@ require (
k8s.io/apimachinery v0.22.1
k8s.io/client-go v0.22.1
sigs.k8s.io/controller-runtime v0.10.0
sigs.k8s.io/gateway-api v0.3.0
sigs.k8s.io/gateway-api v0.4.1
sigs.k8s.io/yaml v1.2.0
)
Loading

0 comments on commit 74180e6

Please sign in to comment.