-
Notifications
You must be signed in to change notification settings - Fork 13
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(metagen): client_py
#802
Merged
Yohe-Am
merged 17 commits into
feat/MET-609/clien-ts-gen
from
feat/MET-567/python-gen-client
Aug 7, 2024
Merged
feat(metagen): client_py
#802
Yohe-Am
merged 17 commits into
feat/MET-609/clien-ts-gen
from
feat/MET-567/python-gen-client
Aug 7, 2024
Conversation
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<!-- Pull requests are squashed and merged using: - their title as the commit message - their description as the commit body Having a good title and description is important for the users to get readable changelog. --> <!-- 1. Explain WHAT the change is about --> - Gleap.io was removed a while back - this adds it back so visitors can open ticket and suggest feedback - internally, we will use this to fine tune the documentation
- Bumps metatype version to 0.4.5 - Bumps ghjk to latest commit - Fixes `setup` whiz task to avoid issues on macos - Fixes release pipeline to publish JSR MET-614 MET-606 MET-605 MET-613 #### Migration notes _No changes required._ - [ ] The change comes with new or modified tests - [ ] Hard-to-understand functions have explanatory comments - [ ] End-user documentation is updated to reflect the change
## Improve the documentation on `quick-start` page - [x] add dev hunt result to homepage. <!-- Pull requests are squashed and merged using: - their title as the commit message - their description as the commit body Having a good title and description is important for the users to get readable changelog. --> <!-- 1. Explain WHAT the change is about --> - <!-- 2. Explain WHY the change cannot be made simpler --> - <!-- 3. Explain HOW users should update their code --> #### Migration notes ... - [ ] The change comes with new or modified tests - [ ] Hard-to-understand functions have explanatory comments - [ ] End-user documentation is updated to reflect the change
<!-- Pull requests are squashed and merged using: - their title as the commit message - their description as the commit body Having a good title and description is important for the users to get readable changelog. --> <!-- 1. Explain WHAT the change is about --> - <!-- 2. Explain WHY the change cannot be made simpler --> - <!-- 3. Explain HOW users should update their code --> #### Migration notes ... - [ ] The change comes with new or modified tests - [ ] Hard-to-understand functions have explanatory comments - [ ] End-user documentation is updated to reflect the change
- Bump version to 0.4.6-0 - Add sanity tests for published SDKs - Bump deno to 1.45.2 - Bump rust to 1.79.0 - Fix myriad of bugs #### Migration notes ... - [x] The change comes with new or modified tests - [ ] Hard-to-understand functions have explanatory comments - [ ] End-user documentation is updated to reflect the change <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit - **New Features** - Introduced new logging capabilities in the `ConnectedEngine` with adjustable logging levels. - Implemented cleanup procedures in tests to enhance resource management. - **Bug Fixes** - Fixed import paths for permissions to ensure correct functionality in tests and applications. - **Version Updates** - Incremented version numbers across multiple projects and packages to reflect ongoing development and improvements. - **Documentation** - Added comments to clarify code behavior and potential future considerations in various modules. - **Refactor** - Optimized string handling in several functions and adjusted method signatures for improved clarity and efficiency. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## feat/MET-609/clien-ts-gen #802 +/- ##
=============================================================
- Coverage 69.91% 69.56% -0.35%
=============================================================
Files 139 142 +3
Lines 16181 16473 +292
Branches 1475 1473 -2
=============================================================
+ Hits 11313 11460 +147
- Misses 4843 4988 +145
Partials 25 25 ☔ View full report in Codecov by Sentry. |
Generated code.# This file was @generated by metagen and is intended
# to be generated again on subsequent metagen runs.
import typing
import dataclasses as dc
import json
import urllib.request as request
import urllib.error
import http.client as http_c
@dc.dataclass
class NodeArgValue:
type_name: str
value: typing.Any
NodeArgs = typing.Dict[str, NodeArgValue]
Out = typing.TypeVar("Out", covariant=True)
@dc.dataclass
class SelectNode(typing.Generic[Out]):
name: str
args: typing.Union[NodeArgs, None]
sub_nodes: typing.Union[typing.List["SelectNode"], None]
_phantom: typing.Union[None, Out] = None
@dc.dataclass
class QueryNode(typing.Generic[Out], SelectNode[Out]):
pass
@dc.dataclass
class MutationNode(typing.Generic[Out], SelectNode[Out]):
pass
ArgT = typing.TypeVar("ArgT")
SelectionT = typing.TypeVar("SelectionT")
AliasInfo = typing.Dict[str, SelectionT]
ScalarSelectNoArgs = typing.Union[bool, None] # | AliasInfo['ScalarSelectNoArgs'];
ScalarSelectArgs = typing.Union[
ArgT, typing.Literal[False], None
] # | AliasInfo['ScalarSelectArgs'];
CompositeSelectNoArgs = typing.Union[
SelectionT, typing.Literal[False], None
] # | AliasInfo['CompositSelectNoArgs'];
CompositeSelectArgs = typing.Union[
typing.Tuple[ArgT, SelectionT], typing.Literal[False], None
] # | AliasInfo['CompositSelectArgs'];
@dc.dataclass
class SelectionFlags:
select_all: typing.Union[bool, None] = None
class Selection(typing.TypedDict, total=False):
_: SelectionFlags
SelectionGeneric = typing.Dict[
str,
typing.Union[
SelectionFlags,
ScalarSelectNoArgs,
ScalarSelectArgs[typing.Mapping[str, typing.Any]],
CompositeSelectNoArgs,
CompositeSelectArgs[typing.Mapping[str, typing.Any], typing.Any],
],
]
@dc.dataclass
class NodeMeta:
sub_nodes: typing.Union[typing.Dict[str, "NodeMeta"], None] = None
arg_types: typing.Union[typing.Dict[str, str], None] = None
def selection_to_nodes(
selection: SelectionGeneric, metas: typing.Dict[str, NodeMeta], parent_path: str
) -> typing.List[SelectNode[typing.Any]]:
out = []
flags = selection.get("_")
if flags is not None and not isinstance(flags, SelectionFlags):
raise Exception(
f"selection field '_' should be of type SelectionFlags but found {type(flags)}"
)
select_all = True if flags is not None and flags.select_all else False
found_nodes = set(selection.keys())
for node_name, meta in metas.items():
found_nodes.remove(node_name)
node_selection = selection[node_name]
if (node_selection is None and not select_all) or not node_selection:
# this node was not selected
continue
node_args: typing.Union[NodeArgs, None] = None
if meta.arg_types is not None:
if not isinstance(node_selection, tuple):
raise Exception(
f"node at {parent_path}.{node_name} is a scalar that "
+ "requires arguments "
+ f"but selection is typeof {type(node_selection)}"
)
arg = node_selection[0]
if not isinstance(arg, dict):
raise Exception(
f"node at {parent_path}.{node_name} is a scalar that "
+ "requires argument object "
+ f"but first element of selection is typeof {type(node_selection)}"
)
expected_args = {key: val for key, val in meta.arg_types.items()}
node_args = {}
for key, val in arg.items():
ty_name = expected_args.pop(key)
if ty_name is None:
raise Exception(
f"unexpected argument ${key} at {parent_path}.{node_name}"
)
node_args[key] = NodeArgValue(ty_name, val)
sub_nodes: typing.Union[typing.List[SelectNode], None] = None
if meta.sub_nodes is not None:
sub_selections = node_selection
if meta.arg_types is not None:
if not isinstance(node_selection, tuple):
raise Exception(
f"node at {parent_path}.{node_name} is a composite "
+ "requires argument object "
+ f"but selection is typeof {type(node_selection)}"
)
sub_selections = node_selection[1]
elif isinstance(sub_selections, tuple):
raise Exception(
f"node at {parent_path}.{node_selection} "
+ "is a composite that takes no arguments "
+ f"but selection is typeof {type(node_selection)}",
)
if not isinstance(sub_selections, dict):
raise Exception(
f"node at {parent_path}.{node_name} "
+ "is a no argument composite but first element of "
+ f"selection is typeof {type(node_selection)}",
)
sub_nodes = selection_to_nodes(
sub_selections, meta.sub_nodes, f"{parent_path}.{node_name}"
)
node = SelectNode(node_name, node_args, sub_nodes)
out.append(node)
found_nodes.discard("_")
if len(found_nodes) > 0:
raise Exception(
f"unexpected nodes found in selection set at {parent_path}: {found_nodes}",
)
return out
def convert_query_node_gql(
node: SelectNode,
variables: typing.Dict[str, NodeArgValue],
):
out = node.name
if node.args is not None:
arg_row = ""
for key, val in node.args.items():
name = f"in{len(variables)}"
variables[name] = val
arg_row += f"{key}: ${name}, "
out += f"({arg_row[:-2]})"
if node.sub_nodes is not None:
sub_node_list = ""
for node in node.sub_nodes:
sub_node_list += f"{convert_query_node_gql(node, variables)} "
out += f" {{ {sub_node_list}}}"
return out
@dc.dataclass
class GraphQLTransportOptions:
headers: typing.Dict[str, str]
@dc.dataclass
class GraphQLRequest:
addr: str
method: str
headers: typing.Dict[str, str]
body: bytes
@dc.dataclass
class GraphQLResponse:
req: GraphQLRequest
status: int
headers: typing.Dict[str, str]
body: bytes
class GraphQLTransportBase:
def __init__(
self,
addr: str,
opts: GraphQLTransportOptions,
ty_to_gql_ty_map: typing.Dict[str, str],
):
self.addr = addr
self.opts = opts
self.ty_to_gql_ty_map = ty_to_gql_ty_map
def build_gql(
self,
query: typing.Dict[str, SelectNode],
ty: typing.Union[typing.Literal["query"], typing.Literal["mutation"]],
name: str = "",
):
variables: typing.Dict[str, NodeArgValue] = {}
root_nodes = ""
for key, node in query.items():
root_nodes += f" {key}: {convert_query_node_gql(node, variables)}\n"
args_row = ""
for key, val in variables.items():
args_row += f"${key}: {self.ty_to_gql_ty_map[val.type_name]}, "
doc = f"{ty} {name}({args_row[:-2]}) {{\n{root_nodes}}}"
return (doc, {key: val.value for key, val in variables.items()})
def build_req(
self,
doc: str,
variables: typing.Dict[str, typing.Any],
opts: typing.Union[GraphQLTransportOptions, None] = None,
):
headers = {}
headers.update(self.opts.headers)
if opts:
headers.update(opts.headers)
headers.update(
{
"accept": "application/json",
"content-type": "application/json",
}
)
data = json.dumps({"query": doc, "variables": variables}).encode("utf-8")
return GraphQLRequest(
addr=self.addr,
method="POST",
headers=headers,
body=data,
)
def handle_response(self, res: GraphQLResponse):
if res.status != 200:
raise Exception(f"graphql request failed with status {res.status}", res)
if res.headers.get("content-type") != "application/json":
raise Exception("unexpected content-type in graphql response", res)
parsed = json.loads(res.body)
if parsed.get("errors"):
raise Exception("graphql errors in response", parsed)
return parsed["data"]
class GraphQLTransportUrlib(GraphQLTransportBase):
def fetch(
self,
doc: str,
variables: typing.Dict[str, typing.Any],
opts: typing.Union[GraphQLTransportOptions, None],
):
req = self.build_req(doc, variables, opts)
try:
with request.urlopen(
request.Request(
url=req.addr, method=req.method, headers=req.headers, data=req.body
)
) as res:
http_res: http_c.HTTPResponse = res
return self.handle_response(
GraphQLResponse(
req,
status=http_res.status,
body=http_res.read(),
headers={key: val for key, val in http_res.headers.items()},
)
)
except request.HTTPError as res:
return self.handle_response(
GraphQLResponse(
req,
status=res.status or 599,
body=res.read(),
headers={key: val for key, val in res.headers.items()},
)
)
except urllib.error.URLError as err:
raise Exception(f"URL error: {err.reason}")
def query(
self,
inp: typing.Dict[str, QueryNode[Out]],
opts: typing.Union[GraphQLTransportOptions, None] = None,
) -> typing.Dict[str, Out]:
doc, variables = self.build_gql({key: val for key, val in inp.items()}, "query")
out = self.fetch(doc, variables, opts)
return out
def mutation(
self,
inp: typing.Dict[str, MutationNode[Out]],
opts: typing.Union[GraphQLTransportOptions, None] = None,
) -> typing.Dict[str, Out]:
doc, variables = self.build_gql(
{key: val for key, val in inp.items()}, "mutation"
)
out = self.fetch(doc, variables, opts)
return out
# def queryT[Out](
# self, inp: typing.Tuple[QueryNode[Out, typing.Any, typing.Any], *QueryNode[Out, typing.Any, typing.Any]]
# ) -> typing.Tuple[*Out]:
# return ()
# def prepare_query[Args, K, Out](
# self,
# argType: type[Args],
# inp: Callable[[Args], typing.Dict[K, SelectNode[Out, typing.Any, typing.Any]]],
# ) -> PreparedRequest[Args, K, Out]:
# return PreparedRequest(inp)
class PreparedRequest(typing.Generic[ArgT, Out]):
def __init__(self, inp: typing.Callable[[ArgT], typing.Dict[str, SelectNode[Out]]]):
self.inp = inp
pass
def do(self, args: ArgT) -> typing.Dict[str, Out]:
return {}
class QueryGraphBase:
def __init__(self, ty_to_gql_ty_map: typing.Dict[str, str]):
self.ty_to_gql_ty_map = ty_to_gql_ty_map
def graphql_sync(
self, addr: str, opts: typing.Union[GraphQLTransportOptions, None] = None
):
return GraphQLTransportUrlib(
addr, opts or GraphQLTransportOptions({}), self.ty_to_gql_ty_map
)
# - - - - - - - - - -- - - - - - - -- - - #
class NodeDescs:
@staticmethod
def scalar():
return NodeMeta()
@staticmethod
def Post():
return NodeMeta(
sub_nodes={
"slug": NodeDescs.scalar(),
"title": NodeDescs.scalar(),
},
)
@staticmethod
def Func9():
return NodeMeta(
sub_nodes=NodeDescs.Post().sub_nodes,
arg_types={
"filter": "Optional4",
},
)
@staticmethod
def User():
return NodeMeta(
sub_nodes={
"id": NodeDescs.scalar(),
"email": NodeDescs.scalar(),
"posts": NodeDescs.Func9(),
},
)
@staticmethod
def Func19():
return NodeMeta(
sub_nodes=NodeDescs.User().sub_nodes,
arg_types={
"id": "String13",
},
)
@staticmethod
def Func20():
return NodeMeta(
sub_nodes=NodeDescs.User().sub_nodes,
)
StringUuid = str
StringEmail = str
class User(typing.TypedDict):
id: StringUuid
email: StringEmail
posts: None
class GetUserInput(typing.TypedDict):
id: str
class Post(typing.TypedDict):
slug: str
title: str
Post7 = typing.List[Post]
class GetPostsInput(typing.TypedDict):
filter: typing.Union[str, None]
class PostSelections(Selection, total=False):
slug: ScalarSelectNoArgs
title: ScalarSelectNoArgs
class UserSelections(Selection, total=False):
id: ScalarSelectNoArgs
email: ScalarSelectNoArgs
posts: CompositeSelectArgs[GetPostsInput, PostSelections]
class QueryGraph(QueryGraphBase):
def __init__(self):
self.ty_to_gql_ty_map = {
"String13": "Any",
"Optional4": "Any",
}
def get_user(self, args: GetUserInput, select: UserSelections) -> QueryNode[User]:
node = selection_to_nodes(
{"getUser": (args, select)}, {"getUser": NodeDescs.Func19()}, "$q"
)[0]
return QueryNode(name=node.name, args=node.args, sub_nodes=node.sub_nodes)
def get_posts(
self, args: GetPostsInput, select: PostSelections
) -> QueryNode[Post7]:
node = selection_to_nodes(
{"getPosts": (args, select)}, {"getPosts": NodeDescs.Func9()}, "$q"
)[0]
return QueryNode(name=node.name, args=node.args, sub_nodes=node.sub_nodes)
def no_args(self, select: UserSelections) -> QueryNode[User]:
node = selection_to_nodes(
{"noArgs": select}, {"noArgs": NodeDescs.Func20()}, "$q"
)[0]
return QueryNode(name=node.name, args=node.args, sub_nodes=node.sub_nodes) Sample usage.from client import (
QueryGraph,
GetUserInput,
UserSelections,
GetPostsInput,
PostSelections,
)
import json
import os
qg = QueryGraph()
port = os.getenv("TG_PORT")
gql_client = qg.graphql_sync(f"http://localhost:{port}/sample")
res = gql_client.query(
{
"user": qg.get_user(
GetUserInput(id="1234"),
UserSelections(
id=True,
email=True,
posts=(
GetPostsInput(filter="top"),
PostSelections(slug=True, title=True),
),
),
),
"posts": qg.get_posts(
GetPostsInput(filter="today"), PostSelections(slug=True, title=True)
),
}
)
out: User = out["user"] |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
client_py
as described in docs:/docs/reference/typegraph/client
#777This is a stacked PR on top of #790.
Migration notes
...