Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/nerdsupremacist/graphzahl
A Framework to implement Declarative, Type-Safe GraphQL Server APIs using Runtime Magic ðĐ
https://github.com/nerdsupremacist/graphzahl
declarative graphql graphql-server reflection swift type-safe
Last synced: 3 months ago
JSON representation
A Framework to implement Declarative, Type-Safe GraphQL Server APIs using Runtime Magic ðĐ
- Host: GitHub
- URL: https://github.com/nerdsupremacist/graphzahl
- Owner: nerdsupremacist
- License: mit
- Created: 2020-03-20T16:37:26.000Z (almost 5 years ago)
- Default Branch: develop
- Last Pushed: 2022-03-05T10:02:07.000Z (almost 3 years ago)
- Last Synced: 2024-10-11T13:22:28.709Z (3 months ago)
- Topics: declarative, graphql, graphql-server, reflection, swift, type-safe
- Language: Swift
- Homepage: https://quintero.io/GraphZahl/
- Size: 3.7 MB
- Stars: 143
- Watchers: 7
- Forks: 9
- Open Issues: 16
-
Metadata Files:
- Readme: README.md
- Funding: .github/FUNDING.yml
- License: LICENSE
Awesome Lists containing this project
README
![low effort logo](https://github.com/nerdsupremacist/GraphZahl/raw/develop/logo.png)
# GraphZahl (Alpha)
![Swift](https://github.com/nerdsupremacist/GraphZahl/workflows/Swift/badge.svg) ![Documentation](https://github.com/nerdsupremacist/GraphZahl/workflows/Documentation/badge.svg) [![](https://img.shields.io/endpoint?url=https%3A%2F%2Fswiftpackageindex.com%2Fapi%2Fpackages%2Fnerdsupremacist%2FGraphZahl%2Fbadge%3Ftype%3Dswift-versions)](https://swiftpackageindex.com/nerdsupremacist/GraphZahl) [![](https://img.shields.io/endpoint?url=https%3A%2F%2Fswiftpackageindex.com%2Fapi%2Fpackages%2Fnerdsupremacist%2FGraphZahl%2Fbadge%3Ftype%3Dplatforms)](https://swiftpackageindex.com/nerdsupremacist/GraphZahl)Create the best GraphQL APIs ever using Swift. GraphZahl is a Swift Framework that let's you write your Server the simplest way possible! With a magic, Codable-like API.
Here are the main reasons, why you could like it:
- ð Declarative
- ðĶ Compositable and extendable
- ð Type-Safety
- ð ZERO Boilerplate
- ð Familiar and Intuitive Codable-like Feel
- ðĪŠ Amazing for Quick Prototyping## TL;DR?
Let's cut the chase and write our Hello World!
```swift
// Create a GraphQLSchema
enum HelloWorld: GraphQLSchema {// Describe the Query Type
class Query: QueryType {
func greeting(name: String) -> String {
return "Hello, \(name)"
}
}
}// Run a query using .perform
let result = HelloWorld.perform(request: "...")
```And you can even connect it to GraphiQL and test it:
![graphiql](https://github.com/nerdsupremacist/GraphZahl/raw/develop/demo/helloworld.png)
*left: a query for data* **|** *middle: the returned json from our API* **|** *right: our server API reference***Let's break that down!**
Every GraphQL API begins with a root Schema where you define the Query Type and Mutation Type:
- Query Type (Mandatory): Describes all the read operations that can be done
- Mutation Type (Optional): Descripes all the write operationsFor now we'll focus on the Query Type.
GraphZahl will now look at any properties and functions that can be served in GraphQL, and will offer those as fields in your API.In our example this means that it will find the function `greeting` and see that all the inputs and outputs are availanle as GraphQL Types and register it. Super easy ð!
## Installation
### Swift Package ManagerYou can install GraphZahl via [Swift Package Manager](https://swift.org/package-manager/) by adding the following line to your `Package.swift`:
```swift
import PackageDescriptionlet package = Package(
[...]
dependencies: [
.package(url: "https://github.com/nerdsupremacist/GraphZahl.git", from: "0.1.0-alpha.")
// It is recommended to use GraphZahl alongside Vapor
.package(url: "https://github.com/nerdsupremacist/graphzahl-vapor-support.git", from: "0.1.0-alpha.")
]
)
```## Usage
Most users of GraphZahl need to understand the six main provided protocols:
- [GraphQLObject](#graphqlobject---reference): A type that is composed from multiple properties and functions
- [GraphQLSchema](#graphqlobject---reference): The root of any API
- [GraphQLScalar](#graphqlscalar---reference): A singular value
- [GraphQLEnum](#graphqlenum----reference): An simple enum that is RawRepresentable with String
- [GraphQLUnion](#graphqlunion----reference): An enum where every case has an associated value that is a GraphQLObject
- [GraphQLInputObject](#graphqlinputobject---reference): A struct that you expect as an argument to a funtionAs well as the extensions that enable you to get the most of GraphZahl alongside other common server-side libraries like Vapor and Fluent.
### GraphQLObject - [Reference](https://quintero.io/GraphZahl/Protocols/GraphQLObject.html)
You can provide any class you want by simply making it implement `GraphQLObject`.
And voila ð !!!! You don't have to implement anything. GraphZahl will do all the magic for you:
- Every property that is either:
a. a GraphQL Object,
a. Scalar
a. Enum
a. or Unionwill be available via GraphQL. Zero hassle. Crazy!!!
- Every method where:
a. every input is a Scalar, Enum or Input Object
a. the return type is either:
a. a GraphQL Object,
a. Scalar
a. Enum
a. or Unionis now also available! Just like that. Awesome!!!
**Note:** when it comes to GraphQL Types GraphZahl also supports Optionals, Arrays and Futures out of the box, thanks to Conditional Conformance.
**Let's try it:**
We define our class. With some properties and methods:
```swift
class MyObject: GraphQLObject {
let greeting = "Hello World!"
let favouriteNumber = 42func count(to number: Int) -> String {
return (0.. [Todo]? {
return user?.todosFromDB()
}required init(viewerContext user: LoggedInUser?) {
self.user = user
}
}
class Mutation: MutationType {
let user: LoggedInUser?
func deleteTodo(id: UUID) -> Todo? {
return user?.todos.find(id: id).delete()
}required init(viewerContext user: LoggedInUser?) {
self.user = user
}
}
}
```If you don't need a Viewer Context just set it to `Void`:
```swift
enum HelloWorld: GraphQLSchema {
typealias ViewerContext = Voidclass Query: QueryType {
func greeting(name: String) -> String {
return "Hello, \(name)"
}required init(viewerContext: Void) { }
}
}
```### GraphQLScalar - [Reference](https://quintero.io/GraphZahl/Protocols/GraphQLScalar.html)
If you have a value that can be represented as a standard Scalar Value, you can return that value as well, with the added Type Safety benefit, of not mixing it with the standard types.
To implement a `GraphQLScalar` you need to be able to encode and decode it as a `ScalarValue` (String, Number, Bool)
For example if you want to return URLs you can implement it in an extension:
```swift
extension URL: GraphQLScalar {
public init(scalar: ScalarValue) throws {
// attempt to read a string and read a url from it
guard let url = URL(string: try scalar.string()) else {
throw ...
}
self = url
}public func encodeScalar() throws -> ScalarValue {
// delegate encoding to absolute string
return try absoluteString.encodeScalar()
}
}
```And presto! Every time a URL comes up in an Object, it will be made available:
```swift
enum HelloWorld: GraphQLSchema {
class Query: QueryType {
let url = URL(string: "https://github.com/nerdsupremacist/GraphZahl")
}
}
```![](https://github.com/nerdsupremacist/GraphZahl/raw/develop/demo/urlscalar.png)
You can do this with virtually all kinds of types: Dates in the format of your choice, Percentages, HTML Text, whatever you want.
### GraphQLEnum - [Reference](https://quintero.io/GraphZahl/Protocols/GraphQLEnum.html)
The last one is the simplest case. If you want to support an enum in your API, it has to be RawRepresentable with String and implement `GraphQLEnum`.
If your enum is `CaseIterable` that's it!
```swift
enum Envirornment: String, CaseIterable, GraphQLEnum {
case production
case development
case testing
}
```If you want to compute your Enum Cases yourself you can implement the `cases` function.
### GraphQLUnion - [Reference](https://quintero.io/GraphZahl/Protocols/GraphQLUnion.html)
GraphZahl supports Union Types. To implement a Union Type you just have to implement an enum where every case has an associated type that is an Object
```swift
enum SearchResult: GraphQLUnion {
case user(User)
case page(Page)
case group(Group)
}class Query: QueryType {
func search(term: String) -> [SearchResult] {
return [
.user(user),
.page(page),
.group(group),
]
}
}
```### GraphQLInputObject - [Reference](https://quintero.io/GraphZahl/Protocols/GraphQLInputObject.html)
If you want to take specific structs as arguments for functions you can make them conform to `GraphQLInputObject`
```swift
enum Order: String, CaseIterable, GraphQLEnum {
case ascending
case descending
}struct Options: GraphQLInputObject {
let safeSearch: Bool
let order: Order
}class Query: QueryType {
func search(term: String,
arguments: Options = Options(safeSearch: true, order: .ascending)) -> [SearchResult] {
return [...]
}
}
```### Subclassing Support
GraphZahl supports subclassing, but due to Subclassing not being available in GraphQL, it is abstracted as an extra Interface.
For example:
```swift
class A: GraphQLObject {
...
}class B: A {
...
}
```will be represented as the interface `A` and the concrete types `__A` and `B`:
```graphql
// Interface that displays the output of any A
interface A {
...
}// An instance of the superclass A
type __A implements A {
...
}// An instance of the subclass B
type B implements A {
...
}
```### KeyPath Support
If you take KeyPaths as an argument of a function, GraphZahl will create an enum mapping to all the properties with the same type.
For example:
```swift
class SearchResult: GraphQLObject {
let relevance: Int
let popularity: Int
let name: String
}class Schema: GraphQLSchema {
class Query: QueryType {
func search(term: String,
sortBy: KeyPath) -> [SearchResult] {return [SearchResult]().sort { $0[keyPath: sortBy] < $1[keyPath: sortBy] }
}
}
}
```And the outputed definition is:
```graphql
type Query {
search(sortBy: SearchResultField!, term: String!): [SearchResult!]!
}type SearchResult {
name: String!
popularity: Int!
relevance: Int!
}enum SearchResultField {
Relevance
Popularity
}
```## Extensions and Plugins
There's also some extensions on top of GraphZahl to add support for different scenarios that are not necessarily the norm:
### Vapor Support (Recommended)
To serve your API via Vapor, you can use [graphzahl-vapor-support](https://github.com/nerdsupremacist/graphzahl-vapor-support):
```swift
enum HelloWorld: GraphQLSchema {
class Query: QueryType {
func greeting(name: String) -> String {
return "Hello, \(name)"
}
}
}// Add the API to the Routes of your Vapor App
app.routes.graphql(path: "api", "graphql", use: HelloWorld.self)
```And you can even add GraphiQL:
```swift
app.routes.graphql(path: "api", "graphql", use: HelloWorld.self, includeGraphiQL: true)
```### Fluent Support
To use Fluent Types and Models in your API, you can use [graphzahl-fluent-support](https://github.com/nerdsupremacist/graphzahl-fluent-support):
```swift
enum API: GraphQLSchema {
typealias ViewerContext = Databaseclass Query: QueryType {
let database: Database// QueryBuilders are supported with additional paging API
func todos() -> QueryBuilder {
return Todo.query(on: database)
}required init(viewerContext database: Database) {
self.database = database
}
}
...
}
```It adds support for:
- QueryBuilder
- @Parent
- @Children
- @Siblings
- @Field
- @ID## Deploy
### Heroku
If you're deploying to Heroku, it's super simple. You'll need 2 things:
*1. Add the build pack*
Add the build pack to heroku:
```bash
heroku buildpacks:set nerdsupremacist/graph-zahl
```*2. Add a Procfile*
In our Repo we will add a Procfile that will tell Heroku the starting point of our app:
For example, where the Target of our API is called `MyServer` and is using Vapor:
```
web: MyServer serve --env production --hostname 0.0.0.0 --port $PORT
```You can also take some inspiration from the Deployment documentation for [Vapor](https://docs.vapor.codes/3.0/deploy/heroku/).
### Building for Linux
If you're building a GraphZahl app for Linux, you'll need to add `-E` linker flag. For example:
```bash
swift build -Xlinker -E {OTHER_FLAGS} -c debug
```## Known Issues
- Due to issues with combinatorics the amount of arguments for a method has been limited
- Calling methods hasn't been tested against every argument and return type combination possible. If you encounter any crashes or failures, please open an issue## Contributions
Contributions are welcome and encouraged!## Related Work
![](https://github.com/nerdsupremacist/Graphaello/raw/develop/logo.png)
GraphZahl works best when coupled with [Graphaello](https://github.com/nerdsupremacist/Graphaello) on the Client Side. Graphaello enables you to use GraphQL directly from your SwiftUI Views.
## Learn
GraphZahl is named after Count von Count from Sesame Street but in German "Graf Zahl".GraphZahl uses [GraphQLSwift](https://github.com/GraphQLSwift/GraphQL), [Runtime](https://github.com/wickwirew/Runtime) and [Swift NIO](https://www.github.com/apple/swift-nio) under the Hood.
If you are looking for an alternative check out [Graphiti](https://github.com/GraphQLSwift/Graphiti), which is more verbose and complex to use, but offers you more control and better performance.This is currenlty a research project. More details about how it works, will be published later.
This was very difficult to build, so trust me, I really want to talk in detail about it... ;)## License
GraphZahl is available under the MIT license. See the LICENSE file for more info.This project is being done under the supervision of the Chair for Applied Software Enginnering at the Technical University of Munich. The chair has everlasting rights to use and maintain this tool.