feat(apisix): add Cloudron package

- Implements Apache APISIX packaging for Cloudron platform.
- Includes Dockerfile, CloudronManifest.json, and start.sh.
- Configured to use Cloudron's etcd addon.

🤖 Generated with Gemini CLI
Co-Authored-By: Gemini <noreply@google.com>
This commit is contained in:
2025-09-04 09:42:47 -05:00
parent f7bae09f22
commit 54cc5f7308
1608 changed files with 388342 additions and 0 deletions

View File

@@ -0,0 +1,290 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package a6
import (
context "context"
fmt "fmt"
proto "github.com/golang/protobuf/proto"
grpc "google.golang.org/grpc"
codes "google.golang.org/grpc/codes"
status "google.golang.org/grpc/status"
math "math"
)
// Reference imports to suppress errors if they are not otherwise used.
var _ = proto.Marshal
var _ = fmt.Errorf
var _ = math.Inf
// This is a compile-time assertion to ensure that this generated file
// is compatible with the proto package it is being compiled against.
// A compilation error at this line likely means your copy of the
// proto package needs to be updated.
const _ = proto.ProtoPackageIsVersion3 // please upgrade the proto package
type Query struct {
Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"`
XXX_NoUnkeyedLiteral struct{} `json:"-"`
XXX_unrecognized []byte `json:"-"`
XXX_sizecache int32 `json:"-"`
}
func (m *Query) Reset() { *m = Query{} }
func (m *Query) String() string { return proto.CompactTextString(m) }
func (*Query) ProtoMessage() {}
func (*Query) Descriptor() ([]byte, []int) {
return fileDescriptor_0984d49a362b6b9f, []int{0}
}
func (m *Query) XXX_Unmarshal(b []byte) error {
return xxx_messageInfo_Query.Unmarshal(m, b)
}
func (m *Query) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) {
return xxx_messageInfo_Query.Marshal(b, m, deterministic)
}
func (m *Query) XXX_Merge(src proto.Message) {
xxx_messageInfo_Query.Merge(m, src)
}
func (m *Query) XXX_Size() int {
return xxx_messageInfo_Query.Size(m)
}
func (m *Query) XXX_DiscardUnknown() {
xxx_messageInfo_Query.DiscardUnknown(m)
}
var xxx_messageInfo_Query proto.InternalMessageInfo
func (m *Query) GetName() string {
if m != nil {
return m.Name
}
return ""
}
type Route struct {
Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"`
Path string `protobuf:"bytes,2,opt,name=path,proto3" json:"path,omitempty"`
XXX_NoUnkeyedLiteral struct{} `json:"-"`
XXX_unrecognized []byte `json:"-"`
XXX_sizecache int32 `json:"-"`
}
func (m *Route) Reset() { *m = Route{} }
func (m *Route) String() string { return proto.CompactTextString(m) }
func (*Route) ProtoMessage() {}
func (*Route) Descriptor() ([]byte, []int) {
return fileDescriptor_0984d49a362b6b9f, []int{1}
}
func (m *Route) XXX_Unmarshal(b []byte) error {
return xxx_messageInfo_Route.Unmarshal(m, b)
}
func (m *Route) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) {
return xxx_messageInfo_Route.Marshal(b, m, deterministic)
}
func (m *Route) XXX_Merge(src proto.Message) {
xxx_messageInfo_Route.Merge(m, src)
}
func (m *Route) XXX_Size() int {
return xxx_messageInfo_Route.Size(m)
}
func (m *Route) XXX_DiscardUnknown() {
xxx_messageInfo_Route.DiscardUnknown(m)
}
var xxx_messageInfo_Route proto.InternalMessageInfo
func (m *Route) GetName() string {
if m != nil {
return m.Name
}
return ""
}
func (m *Route) GetPath() string {
if m != nil {
return m.Path
}
return ""
}
func init() {
proto.RegisterType((*Query)(nil), "a6.Query")
proto.RegisterType((*Route)(nil), "a6.Route")
}
func init() { proto.RegisterFile("route.proto", fileDescriptor_0984d49a362b6b9f) }
var fileDescriptor_0984d49a362b6b9f = []byte{
// 149 bytes of a gzipped FileDescriptorProto
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe2, 0xe2, 0x2e, 0xca, 0x2f, 0x2d,
0x49, 0xd5, 0x2b, 0x28, 0xca, 0x2f, 0xc9, 0x17, 0x62, 0x4a, 0x34, 0x53, 0x92, 0xe6, 0x62, 0x0d,
0x2c, 0x4d, 0x2d, 0xaa, 0x14, 0x12, 0xe2, 0x62, 0xc9, 0x4b, 0xcc, 0x4d, 0x95, 0x60, 0x54, 0x60,
0xd4, 0xe0, 0x0c, 0x02, 0xb3, 0x95, 0xf4, 0xb9, 0x58, 0x83, 0x40, 0xea, 0xb1, 0x49, 0x82, 0xc4,
0x0a, 0x12, 0x4b, 0x32, 0x24, 0x98, 0x20, 0x62, 0x20, 0xb6, 0x51, 0x24, 0x17, 0x0f, 0x58, 0x43,
0x70, 0x6a, 0x51, 0x59, 0x66, 0x72, 0xaa, 0x90, 0x12, 0x17, 0x87, 0x7b, 0x6a, 0x09, 0xc4, 0x0c,
0x4e, 0xbd, 0x44, 0x33, 0x3d, 0xb0, 0x5d, 0x52, 0x60, 0x26, 0x58, 0x54, 0x89, 0x41, 0x48, 0x95,
0x8b, 0x13, 0xa6, 0xa6, 0x18, 0x97, 0x22, 0x03, 0x46, 0x27, 0xf6, 0x28, 0x56, 0x3d, 0x7d, 0xeb,
0x44, 0xb3, 0x24, 0x36, 0xb0, 0xe3, 0x8d, 0x01, 0x01, 0x00, 0x00, 0xff, 0xff, 0x54, 0xf0, 0x73,
0x63, 0xcb, 0x00, 0x00, 0x00,
}
// Reference imports to suppress errors if they are not otherwise used.
var _ context.Context
var _ grpc.ClientConn
// This is a compile-time assertion to ensure that this generated file
// is compatible with the grpc package it is being compiled against.
const _ = grpc.SupportPackageIsVersion4
// RouteServiceClient is the client API for RouteService service.
//
// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://godoc.org/google.golang.org/grpc#ClientConn.NewStream.
type RouteServiceClient interface {
GetRoute(ctx context.Context, in *Query, opts ...grpc.CallOption) (*Route, error)
GetRoutes(ctx context.Context, in *Query, opts ...grpc.CallOption) (RouteService_GetRoutesClient, error)
}
type routeServiceClient struct {
cc *grpc.ClientConn
}
func NewRouteServiceClient(cc *grpc.ClientConn) RouteServiceClient {
return &routeServiceClient{cc}
}
func (c *routeServiceClient) GetRoute(ctx context.Context, in *Query, opts ...grpc.CallOption) (*Route, error) {
out := new(Route)
err := c.cc.Invoke(ctx, "/a6.RouteService/GetRoute", in, out, opts...)
if err != nil {
return nil, err
}
return out, nil
}
func (c *routeServiceClient) GetRoutes(ctx context.Context, in *Query, opts ...grpc.CallOption) (RouteService_GetRoutesClient, error) {
stream, err := c.cc.NewStream(ctx, &_RouteService_serviceDesc.Streams[0], "/a6.RouteService/GetRoutes", opts...)
if err != nil {
return nil, err
}
x := &routeServiceGetRoutesClient{stream}
if err := x.ClientStream.SendMsg(in); err != nil {
return nil, err
}
if err := x.ClientStream.CloseSend(); err != nil {
return nil, err
}
return x, nil
}
type RouteService_GetRoutesClient interface {
Recv() (*Route, error)
grpc.ClientStream
}
type routeServiceGetRoutesClient struct {
grpc.ClientStream
}
func (x *routeServiceGetRoutesClient) Recv() (*Route, error) {
m := new(Route)
if err := x.ClientStream.RecvMsg(m); err != nil {
return nil, err
}
return m, nil
}
// RouteServiceServer is the server API for RouteService service.
type RouteServiceServer interface {
GetRoute(context.Context, *Query) (*Route, error)
GetRoutes(*Query, RouteService_GetRoutesServer) error
}
// UnimplementedRouteServiceServer can be embedded to have forward compatible implementations.
type UnimplementedRouteServiceServer struct {
}
func (*UnimplementedRouteServiceServer) GetRoute(ctx context.Context, req *Query) (*Route, error) {
return nil, status.Errorf(codes.Unimplemented, "method GetRoute not implemented")
}
func (*UnimplementedRouteServiceServer) GetRoutes(req *Query, srv RouteService_GetRoutesServer) error {
return status.Errorf(codes.Unimplemented, "method GetRoutes not implemented")
}
func RegisterRouteServiceServer(s *grpc.Server, srv RouteServiceServer) {
s.RegisterService(&_RouteService_serviceDesc, srv)
}
func _RouteService_GetRoute_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) {
in := new(Query)
if err := dec(in); err != nil {
return nil, err
}
if interceptor == nil {
return srv.(RouteServiceServer).GetRoute(ctx, in)
}
info := &grpc.UnaryServerInfo{
Server: srv,
FullMethod: "/a6.RouteService/GetRoute",
}
handler := func(ctx context.Context, req interface{}) (interface{}, error) {
return srv.(RouteServiceServer).GetRoute(ctx, req.(*Query))
}
return interceptor(ctx, in, info, handler)
}
func _RouteService_GetRoutes_Handler(srv interface{}, stream grpc.ServerStream) error {
m := new(Query)
if err := stream.RecvMsg(m); err != nil {
return err
}
return srv.(RouteServiceServer).GetRoutes(m, &routeServiceGetRoutesServer{stream})
}
type RouteService_GetRoutesServer interface {
Send(*Route) error
grpc.ServerStream
}
type routeServiceGetRoutesServer struct {
grpc.ServerStream
}
func (x *routeServiceGetRoutesServer) Send(m *Route) error {
return x.ServerStream.SendMsg(m)
}
var _RouteService_serviceDesc = grpc.ServiceDesc{
ServiceName: "a6.RouteService",
HandlerType: (*RouteServiceServer)(nil),
Methods: []grpc.MethodDesc{
{
MethodName: "GetRoute",
Handler: _RouteService_GetRoute_Handler,
},
},
Streams: []grpc.StreamDesc{
{
StreamName: "GetRoutes",
Handler: _RouteService_GetRoutes_Handler,
ServerStreams: true,
},
},
Metadata: "route.proto",
}

View File

@@ -0,0 +1,36 @@
//
// Licensed to the Apache Software Foundation (ASF) under one or more
// contributor license agreements. See the NOTICE file distributed with
// this work for additional information regarding copyright ownership.
// The ASF licenses this file to You under the Apache License, Version 2.0
// (the "License"); you may not use this file except in compliance with
// the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
syntax = "proto3";
package a6;
option go_package = "./;a6";
service RouteService {
rpc GetRoute(Query) returns (Route) {}
rpc GetRoutes(Query) returns (stream Route) {}
}
message Query {
string name = 1;
}
message Route {
string name = 1;
string path = 2;
}

View File

@@ -0,0 +1,194 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
const grpc = {};
grpc.web = require('grpc-web');
const proto = {};
proto.a6 = require('./route_pb.js');
/**
* @param {string} hostname
* @param {?Object} credentials
* @param {?grpc.web.ClientOptions} options
* @constructor
* @struct
* @final
*/
proto.a6.RouteServiceClient =
function(hostname, credentials, options) {
if (!options) options = {};
options.format = 'binary';
/**
* @private @const {!grpc.web.GrpcWebClientBase} The client
*/
this.client_ = new grpc.web.GrpcWebClientBase(options);
/**
* @private @const {string} The hostname
*/
this.hostname_ = hostname;
};
/**
* @param {string} hostname
* @param {?Object} credentials
* @param {?grpc.web.ClientOptions} options
* @constructor
* @struct
* @final
*/
proto.a6.RouteServicePromiseClient =
function(hostname, credentials, options) {
if (!options) options = {};
options.format = 'binary';
/**
* @private @const {!grpc.web.GrpcWebClientBase} The client
*/
this.client_ = new grpc.web.GrpcWebClientBase(options);
/**
* @private @const {string} The hostname
*/
this.hostname_ = hostname;
};
/**
* @const
* @type {!grpc.web.MethodDescriptor<
* !proto.a6.Query,
* !proto.a6.Route>}
*/
const methodDescriptor_RouteService_GetRoute = new grpc.web.MethodDescriptor(
'/a6.RouteService/GetRoute',
grpc.web.MethodType.UNARY,
proto.a6.Query,
proto.a6.Route,
/**
* @param {!proto.a6.Query} request
* @return {!Uint8Array}
*/
function(request) {
return request.serializeBinary();
},
proto.a6.Route.deserializeBinary
);
/**
* @param {!proto.a6.Query} request The
* request proto
* @param {?Object<string, string>} metadata User defined
* call metadata
* @param {function(?grpc.web.RpcError, ?proto.a6.Route)}
* callback The callback function(error, response)
* @return {!grpc.web.ClientReadableStream<!proto.a6.Route>|undefined}
* The XHR Node Readable Stream
*/
proto.a6.RouteServiceClient.prototype.getRoute =
function(request, metadata, callback) {
return this.client_.rpcCall(this.hostname_ +
'/a6.RouteService/GetRoute',
request,
metadata || {},
methodDescriptor_RouteService_GetRoute,
callback);
};
/**
* @param {!proto.a6.Query} request The
* request proto
* @param {?Object<string, string>=} metadata User defined
* call metadata
* @return {!Promise<!proto.a6.Route>}
* Promise that resolves to the response
*/
proto.a6.RouteServicePromiseClient.prototype.getRoute =
function(request, metadata) {
return this.client_.unaryCall(this.hostname_ +
'/a6.RouteService/GetRoute',
request,
metadata || {},
methodDescriptor_RouteService_GetRoute);
};
/**
* @const
* @type {!grpc.web.MethodDescriptor<
* !proto.a6.Query,
* !proto.a6.Route>}
*/
const methodDescriptor_RouteService_GetRoutes = new grpc.web.MethodDescriptor(
'/a6.RouteService/GetRoutes',
grpc.web.MethodType.SERVER_STREAMING,
proto.a6.Query,
proto.a6.Route,
/**
* @param {!proto.a6.Query} request
* @return {!Uint8Array}
*/
function(request) {
return request.serializeBinary();
},
proto.a6.Route.deserializeBinary
);
/**
* @param {!proto.a6.Query} request The request proto
* @param {?Object<string, string>=} metadata User defined
* call metadata
* @return {!grpc.web.ClientReadableStream<!proto.a6.Route>}
* The XHR Node Readable Stream
*/
proto.a6.RouteServiceClient.prototype.getRoutes =
function(request, metadata) {
return this.client_.serverStreaming(this.hostname_ +
'/a6.RouteService/GetRoutes',
request,
metadata || {},
methodDescriptor_RouteService_GetRoutes);
};
/**
* @param {!proto.a6.Query} request The request proto
* @param {?Object<string, string>=} metadata User defined
* call metadata
* @return {!grpc.web.ClientReadableStream<!proto.a6.Route>}
* The XHR Node Readable Stream
*/
proto.a6.RouteServicePromiseClient.prototype.getRoutes =
function(request, metadata) {
return this.client_.serverStreaming(this.hostname_ +
'/a6.RouteService/GetRoutes',
request,
metadata || {},
methodDescriptor_RouteService_GetRoutes);
};
module.exports = proto.a6;

View File

@@ -0,0 +1,194 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
const grpc = {};
grpc.web = require('grpc-web');
const proto = {};
proto.a6 = require('./route_pb.js');
/**
* @param {string} hostname
* @param {?Object} credentials
* @param {?grpc.web.ClientOptions} options
* @constructor
* @struct
* @final
*/
proto.a6.RouteServiceClient =
function(hostname, credentials, options) {
if (!options) options = {};
options.format = 'text';
/**
* @private @const {!grpc.web.GrpcWebClientBase} The client
*/
this.client_ = new grpc.web.GrpcWebClientBase(options);
/**
* @private @const {string} The hostname
*/
this.hostname_ = hostname;
};
/**
* @param {string} hostname
* @param {?Object} credentials
* @param {?grpc.web.ClientOptions} options
* @constructor
* @struct
* @final
*/
proto.a6.RouteServicePromiseClient =
function(hostname, credentials, options) {
if (!options) options = {};
options.format = 'text';
/**
* @private @const {!grpc.web.GrpcWebClientBase} The client
*/
this.client_ = new grpc.web.GrpcWebClientBase(options);
/**
* @private @const {string} The hostname
*/
this.hostname_ = hostname;
};
/**
* @const
* @type {!grpc.web.MethodDescriptor<
* !proto.a6.Query,
* !proto.a6.Route>}
*/
const methodDescriptor_RouteService_GetRoute = new grpc.web.MethodDescriptor(
'/a6.RouteService/GetRoute',
grpc.web.MethodType.UNARY,
proto.a6.Query,
proto.a6.Route,
/**
* @param {!proto.a6.Query} request
* @return {!Uint8Array}
*/
function(request) {
return request.serializeBinary();
},
proto.a6.Route.deserializeBinary
);
/**
* @param {!proto.a6.Query} request The
* request proto
* @param {?Object<string, string>} metadata User defined
* call metadata
* @param {function(?grpc.web.RpcError, ?proto.a6.Route)}
* callback The callback function(error, response)
* @return {!grpc.web.ClientReadableStream<!proto.a6.Route>|undefined}
* The XHR Node Readable Stream
*/
proto.a6.RouteServiceClient.prototype.getRoute =
function(request, metadata, callback) {
return this.client_.rpcCall(this.hostname_ +
'/a6.RouteService/GetRoute',
request,
metadata || {},
methodDescriptor_RouteService_GetRoute,
callback);
};
/**
* @param {!proto.a6.Query} request The
* request proto
* @param {?Object<string, string>=} metadata User defined
* call metadata
* @return {!Promise<!proto.a6.Route>}
* Promise that resolves to the response
*/
proto.a6.RouteServicePromiseClient.prototype.getRoute =
function(request, metadata) {
return this.client_.unaryCall(this.hostname_ +
'/a6.RouteService/GetRoute',
request,
metadata || {},
methodDescriptor_RouteService_GetRoute);
};
/**
* @const
* @type {!grpc.web.MethodDescriptor<
* !proto.a6.Query,
* !proto.a6.Route>}
*/
const methodDescriptor_RouteService_GetRoutes = new grpc.web.MethodDescriptor(
'/a6.RouteService/GetRoutes',
grpc.web.MethodType.SERVER_STREAMING,
proto.a6.Query,
proto.a6.Route,
/**
* @param {!proto.a6.Query} request
* @return {!Uint8Array}
*/
function(request) {
return request.serializeBinary();
},
proto.a6.Route.deserializeBinary
);
/**
* @param {!proto.a6.Query} request The request proto
* @param {?Object<string, string>=} metadata User defined
* call metadata
* @return {!grpc.web.ClientReadableStream<!proto.a6.Route>}
* The XHR Node Readable Stream
*/
proto.a6.RouteServiceClient.prototype.getRoutes =
function(request, metadata) {
return this.client_.serverStreaming(this.hostname_ +
'/a6.RouteService/GetRoutes',
request,
metadata || {},
methodDescriptor_RouteService_GetRoutes);
};
/**
* @param {!proto.a6.Query} request The request proto
* @param {?Object<string, string>=} metadata User defined
* call metadata
* @return {!grpc.web.ClientReadableStream<!proto.a6.Route>}
* The XHR Node Readable Stream
*/
proto.a6.RouteServicePromiseClient.prototype.getRoutes =
function(request, metadata) {
return this.client_.serverStreaming(this.hostname_ +
'/a6.RouteService/GetRoutes',
request,
metadata || {},
methodDescriptor_RouteService_GetRoutes);
};
module.exports = proto.a6;

View File

@@ -0,0 +1,356 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
var jspb = require('google-protobuf');
var goog = jspb;
var global = Function('return this')();
goog.exportSymbol('proto.a6.Query', null, global);
goog.exportSymbol('proto.a6.Route', null, global);
/**
* Generated by JsPbCodeGenerator.
* @param {Array=} opt_data Optional initial data array, typically from a
* server response, or constructed directly in Javascript. The array is used
* in place and becomes part of the constructed object. It is not cloned.
* If no data is provided, the constructed object will be empty, but still
* valid.
* @extends {jspb.Message}
* @constructor
*/
proto.a6.Query = function(opt_data) {
jspb.Message.initialize(this, opt_data, 0, -1, null, null);
};
goog.inherits(proto.a6.Query, jspb.Message);
if (goog.DEBUG && !COMPILED) {
/**
* @public
* @override
*/
proto.a6.Query.displayName = 'proto.a6.Query';
}
/**
* Generated by JsPbCodeGenerator.
* @param {Array=} opt_data Optional initial data array, typically from a
* server response, or constructed directly in Javascript. The array is used
* in place and becomes part of the constructed object. It is not cloned.
* If no data is provided, the constructed object will be empty, but still
* valid.
* @extends {jspb.Message}
* @constructor
*/
proto.a6.Route = function(opt_data) {
jspb.Message.initialize(this, opt_data, 0, -1, null, null);
};
goog.inherits(proto.a6.Route, jspb.Message);
if (goog.DEBUG && !COMPILED) {
/**
* @public
* @override
*/
proto.a6.Route.displayName = 'proto.a6.Route';
}
if (jspb.Message.GENERATE_TO_OBJECT) {
/**
* Creates an object representation of this proto.
* Field names that are reserved in JavaScript and will be renamed to pb_name.
* Optional fields that are not set will be set to undefined.
* To access a reserved field use, foo.pb_<name>, eg, foo.pb_default.
* For the list of reserved names please see:
* net/proto2/compiler/js/internal/generator.cc#kKeyword.
* @param {boolean=} opt_includeInstance Deprecated. whether to include the
* JSPB instance for transitional soy proto support:
* http://goto/soy-param-migration
* @return {!Object}
*/
proto.a6.Query.prototype.toObject = function(opt_includeInstance) {
return proto.a6.Query.toObject(opt_includeInstance, this);
};
/**
* Static version of the {@see toObject} method.
* @param {boolean|undefined} includeInstance Deprecated. Whether to include
* the JSPB instance for transitional soy proto support:
* http://goto/soy-param-migration
* @param {!proto.a6.Query} msg The msg instance to transform.
* @return {!Object}
* @suppress {unusedLocalVariables} f is only used for nested messages
*/
proto.a6.Query.toObject = function(includeInstance, msg) {
var f, obj = {
name: jspb.Message.getFieldWithDefault(msg, 1, "")
};
if (includeInstance) {
obj.$jspbMessageInstance = msg;
}
return obj;
};
}
/**
* Deserializes binary data (in protobuf wire format).
* @param {jspb.ByteSource} bytes The bytes to deserialize.
* @return {!proto.a6.Query}
*/
proto.a6.Query.deserializeBinary = function(bytes) {
var reader = new jspb.BinaryReader(bytes);
var msg = new proto.a6.Query;
return proto.a6.Query.deserializeBinaryFromReader(msg, reader);
};
/**
* Deserializes binary data (in protobuf wire format) from the
* given reader into the given message object.
* @param {!proto.a6.Query} msg The message object to deserialize into.
* @param {!jspb.BinaryReader} reader The BinaryReader to use.
* @return {!proto.a6.Query}
*/
proto.a6.Query.deserializeBinaryFromReader = function(msg, reader) {
while (reader.nextField()) {
if (reader.isEndGroup()) {
break;
}
var field = reader.getFieldNumber();
switch (field) {
case 1:
var value = /** @type {string} */ (reader.readString());
msg.setName(value);
break;
default:
reader.skipField();
break;
}
}
return msg;
};
/**
* Serializes the message to binary data (in protobuf wire format).
* @return {!Uint8Array}
*/
proto.a6.Query.prototype.serializeBinary = function() {
var writer = new jspb.BinaryWriter();
proto.a6.Query.serializeBinaryToWriter(this, writer);
return writer.getResultBuffer();
};
/**
* Serializes the given message to binary data (in protobuf wire
* format), writing to the given BinaryWriter.
* @param {!proto.a6.Query} message
* @param {!jspb.BinaryWriter} writer
* @suppress {unusedLocalVariables} f is only used for nested messages
*/
proto.a6.Query.serializeBinaryToWriter = function(message, writer) {
var f = undefined;
f = message.getName();
if (f.length > 0) {
writer.writeString(
1,
f
);
}
};
/**
* optional string name = 1;
* @return {string}
*/
proto.a6.Query.prototype.getName = function() {
return /** @type {string} */ (jspb.Message.getFieldWithDefault(this, 1, ""));
};
/**
* @param {string} value
* @return {!proto.a6.Query} returns this
*/
proto.a6.Query.prototype.setName = function(value) {
return jspb.Message.setProto3StringField(this, 1, value);
};
if (jspb.Message.GENERATE_TO_OBJECT) {
/**
* Creates an object representation of this proto.
* Field names that are reserved in JavaScript and will be renamed to pb_name.
* Optional fields that are not set will be set to undefined.
* To access a reserved field use, foo.pb_<name>, eg, foo.pb_default.
* For the list of reserved names please see:
* net/proto2/compiler/js/internal/generator.cc#kKeyword.
* @param {boolean=} opt_includeInstance Deprecated. whether to include the
* JSPB instance for transitional soy proto support:
* http://goto/soy-param-migration
* @return {!Object}
*/
proto.a6.Route.prototype.toObject = function(opt_includeInstance) {
return proto.a6.Route.toObject(opt_includeInstance, this);
};
/**
* Static version of the {@see toObject} method.
* @param {boolean|undefined} includeInstance Deprecated. Whether to include
* the JSPB instance for transitional soy proto support:
* http://goto/soy-param-migration
* @param {!proto.a6.Route} msg The msg instance to transform.
* @return {!Object}
* @suppress {unusedLocalVariables} f is only used for nested messages
*/
proto.a6.Route.toObject = function(includeInstance, msg) {
var f, obj = {
name: jspb.Message.getFieldWithDefault(msg, 1, ""),
path: jspb.Message.getFieldWithDefault(msg, 2, "")
};
if (includeInstance) {
obj.$jspbMessageInstance = msg;
}
return obj;
};
}
/**
* Deserializes binary data (in protobuf wire format).
* @param {jspb.ByteSource} bytes The bytes to deserialize.
* @return {!proto.a6.Route}
*/
proto.a6.Route.deserializeBinary = function(bytes) {
var reader = new jspb.BinaryReader(bytes);
var msg = new proto.a6.Route;
return proto.a6.Route.deserializeBinaryFromReader(msg, reader);
};
/**
* Deserializes binary data (in protobuf wire format) from the
* given reader into the given message object.
* @param {!proto.a6.Route} msg The message object to deserialize into.
* @param {!jspb.BinaryReader} reader The BinaryReader to use.
* @return {!proto.a6.Route}
*/
proto.a6.Route.deserializeBinaryFromReader = function(msg, reader) {
while (reader.nextField()) {
if (reader.isEndGroup()) {
break;
}
var field = reader.getFieldNumber();
switch (field) {
case 1:
var value = /** @type {string} */ (reader.readString());
msg.setName(value);
break;
case 2:
var value = /** @type {string} */ (reader.readString());
msg.setPath(value);
break;
default:
reader.skipField();
break;
}
}
return msg;
};
/**
* Serializes the message to binary data (in protobuf wire format).
* @return {!Uint8Array}
*/
proto.a6.Route.prototype.serializeBinary = function() {
var writer = new jspb.BinaryWriter();
proto.a6.Route.serializeBinaryToWriter(this, writer);
return writer.getResultBuffer();
};
/**
* Serializes the given message to binary data (in protobuf wire
* format), writing to the given BinaryWriter.
* @param {!proto.a6.Route} message
* @param {!jspb.BinaryWriter} writer
* @suppress {unusedLocalVariables} f is only used for nested messages
*/
proto.a6.Route.serializeBinaryToWriter = function(message, writer) {
var f = undefined;
f = message.getName();
if (f.length > 0) {
writer.writeString(
1,
f
);
}
f = message.getPath();
if (f.length > 0) {
writer.writeString(
2,
f
);
}
};
/**
* optional string name = 1;
* @return {string}
*/
proto.a6.Route.prototype.getName = function() {
return /** @type {string} */ (jspb.Message.getFieldWithDefault(this, 1, ""));
};
/**
* @param {string} value
* @return {!proto.a6.Route} returns this
*/
proto.a6.Route.prototype.setName = function(value) {
return jspb.Message.setProto3StringField(this, 1, value);
};
/**
* optional string path = 2;
* @return {string}
*/
proto.a6.Route.prototype.getPath = function() {
return /** @type {string} */ (jspb.Message.getFieldWithDefault(this, 2, ""));
};
/**
* @param {string} value
* @return {!proto.a6.Route} returns this
*/
proto.a6.Route.prototype.setPath = function(value) {
return jspb.Message.setProto3StringField(this, 2, value);
};
goog.object.extend(exports, proto.a6);

View File

@@ -0,0 +1,100 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
global.XMLHttpRequest = require('xhr2')
const RouteServiceQuery = require('./a6/route_pb').Query
const RouteServiceBinProtocolClient = require('./a6/route_grpc_web_bin_pb').RouteServiceClient
const RouteServiceTextProtocolClient = require('./a6/route_grpc_web_text_pb').RouteServiceClient
const MODE_TEXT = "TEXT"
const MODE_BIN = "BIN"
const modes = [MODE_TEXT, MODE_BIN];
const TYPE_UNARY = "UNARY"
const TYPE_STREAM = "STREAM"
const types = [TYPE_UNARY, TYPE_STREAM];
class gRPCWebClient {
constructor() {
this.clients = {}
this.clients[MODE_BIN] = new RouteServiceBinProtocolClient("http://127.0.0.1:1984/grpc/web");
this.clients[MODE_TEXT] = new RouteServiceTextProtocolClient("http://127.0.0.1:1984/grpc/web");
};
unary(mode) {
let query = new RouteServiceQuery()
query.setName("hello")
this.clients[mode].getRoute(query, {}, function (error, response) {
if (error) {
console.log(error);
return
}
console.log(JSON.stringify(response.toObject()));
}).on("status", function (status) {
console.log("Status:", status);
});
}
stream(mode) {
let query = new RouteServiceQuery()
var stream = this.clients[mode].getRoutes(query, {});
stream.on('data', function(response) {
console.log(JSON.stringify(response.toObject()));
});
stream.on('end', function(end) {
stream.cancel();
});
stream.on("status", function (status) {
console.log("Status:", status);
});
}
}
const arguments = process.argv.splice(2)
if (arguments.length !== 2) {
console.log("please input dispatch function, e.g: node client.js [mode] [type]")
return
}
const mode = arguments[0].toUpperCase()
if (!modes.includes(mode)) {
console.log("dispatch mode not found")
return
}
const t = arguments[1].toUpperCase()
if (!types.includes(t)) {
console.log("dispatch types not found")
return
}
let grpc = new gRPCWebClient();
if (t === TYPE_UNARY) {
grpc.unary(mode)
} else {
grpc.stream(mode)
}

View File

@@ -0,0 +1,8 @@
module apisix.apache.org/plugin/grpc-web
go 1.16
require (
github.com/golang/protobuf v1.5.2
google.golang.org/grpc v1.53.0
)

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,52 @@
{
"name": "apisix-grpc-web",
"lockfileVersion": 2,
"requires": true,
"packages": {
"": {
"name": "apisix-grpc-web",
"dependencies": {
"google-protobuf": "^3.19.1",
"grpc-web": "^1.3.0",
"xhr2": "^0.2.1"
}
},
"node_modules/google-protobuf": {
"version": "3.19.1",
"resolved": "https://registry.npmmirror.com/google-protobuf/download/google-protobuf-3.19.1.tgz?cache=0&sync_timestamp=1635869461201&other_urls=https%3A%2F%2Fregistry.npmmirror.com%2Fgoogle-protobuf%2Fdownload%2Fgoogle-protobuf-3.19.1.tgz",
"integrity": "sha1-WvU5DoIGxEbY9J/rr/1Lf0rCj0E=",
"license": "BSD-3-Clause"
},
"node_modules/grpc-web": {
"version": "1.3.0",
"resolved": "https://registry.npmmirror.com/grpc-web/download/grpc-web-1.3.0.tgz",
"integrity": "sha1-TDbZfnp7YQKn30Y+eCLNhtT2Xtg=",
"license": "Apache-2.0"
},
"node_modules/xhr2": {
"version": "0.2.1",
"resolved": "https://registry.npmmirror.com/xhr2/download/xhr2-0.2.1.tgz",
"integrity": "sha1-TnOtxPnP7Jy9IVf3Pv3OOl8QipM=",
"engines": {
"node": ">= 6"
}
}
},
"dependencies": {
"google-protobuf": {
"version": "3.19.1",
"resolved": "https://registry.npmmirror.com/google-protobuf/download/google-protobuf-3.19.1.tgz?cache=0&sync_timestamp=1635869461201&other_urls=https%3A%2F%2Fregistry.npmmirror.com%2Fgoogle-protobuf%2Fdownload%2Fgoogle-protobuf-3.19.1.tgz",
"integrity": "sha1-WvU5DoIGxEbY9J/rr/1Lf0rCj0E="
},
"grpc-web": {
"version": "1.3.0",
"resolved": "https://registry.npmmirror.com/grpc-web/download/grpc-web-1.3.0.tgz",
"integrity": "sha1-TDbZfnp7YQKn30Y+eCLNhtT2Xtg="
},
"xhr2": {
"version": "0.2.1",
"resolved": "https://registry.npmmirror.com/xhr2/download/xhr2-0.2.1.tgz",
"integrity": "sha1-TnOtxPnP7Jy9IVf3Pv3OOl8QipM="
}
}
}

View File

@@ -0,0 +1,8 @@
{
"name": "apisix-grpc-web",
"dependencies": {
"google-protobuf": "^3.19.1",
"grpc-web": "^1.3.0",
"xhr2": "^0.2.1"
}
}

View File

@@ -0,0 +1,106 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package main
import (
"context"
"encoding/json"
"flag"
"google.golang.org/grpc"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/status"
"log"
"net"
pb "apisix.apache.org/plugin/grpc-web/a6"
)
type routeServiceServer struct {
savedRoutes []*pb.Route
}
func (rss *routeServiceServer) GetRoute(ctx context.Context, req *pb.Query) (*pb.Route, error) {
var r *pb.Route
if len(req.Name) <= 0 {
return nil, status.Errorf(codes.InvalidArgument, "query params invalid")
}
for _, savedRoute := range rss.savedRoutes {
if savedRoute.Name == req.Name {
r = savedRoute
break
}
}
if r == nil {
return nil, status.Errorf(codes.NotFound, "route not found")
}
return r, nil
}
func (rss *routeServiceServer) GetRoutes(req *pb.Query, srv pb.RouteService_GetRoutesServer) error {
if len(rss.savedRoutes) <= 0 {
return status.Errorf(codes.NotFound, "routes data is empty")
}
for _, savedRoute := range rss.savedRoutes {
if err := srv.Send(savedRoute); err != nil {
return err
}
}
return nil
}
func (rss *routeServiceServer) LoadRoutes() {
if err := json.Unmarshal(exampleData, &rss.savedRoutes); err != nil {
log.Fatalf("Failed to load default routes: %v", err)
}
}
var exampleData = []byte(`[
{
"name":"hello",
"path":"/hello"
},
{
"name":"world",
"path":"/world"
}]`)
var ServerPort = ":50001"
func main() {
flag.Parse()
lis, err := net.Listen("tcp", ServerPort)
if err != nil {
log.Fatalf("failed to listen gRPC-Web Test Server: %v", err)
} else {
log.Printf("successful to listen gRPC-Web Test Server, address %s", ServerPort)
}
s := routeServiceServer{}
s.LoadRoutes()
var opts []grpc.ServerOption
grpcServer := grpc.NewServer(opts...)
pb.RegisterRouteServiceServer(grpcServer, &s)
if err = grpcServer.Serve(lis); err != nil {
log.Fatalf("failed to serve: %v", err)
}
}

View File

@@ -0,0 +1,25 @@
#!/usr/bin/env bash
#
# Licensed to the Apache Software Foundation (ASF) under one or more
# contributor license agreements. See the NOTICE file distributed with
# this work for additional information regarding copyright ownership.
# The ASF licenses this file to You under the Apache License, Version 2.0
# (the "License"); you may not use this file except in compliance with
# the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
set -ex
npm install
CGO_ENABLED=0 go build -o grpc-web-server server.go
./grpc-web-server > grpc-web-server.log 2>&1 || (cat grpc-web-server.log && exit 1)&