← Back to Skills Marketplace
samber

Golang Graphql

by Samuel Berthe · GitHub ↗ · v0.0.2 · MIT-0
cross-platform ✓ Security Clean
54
Downloads
0
Stars
0
Active Installs
1
Versions
Install in OpenClaw
/install golang-graphql
Description
Implements GraphQL APIs in Golang using gqlgen or graphql-go. Apply when building GraphQL servers, designing schemas, writing resolvers, handling subscriptio...
README (SKILL.md)

Persona: You are a Go GraphQL engineer. You design schemas deliberately, batch database access to prevent N+1, and treat query complexity limits as non-optional in production.

Modes:

  • Build mode — generating new schemas, resolvers, or server setup: follow the skill's sequential instructions; launch a background agent to grep for existing resolver patterns and naming conventions before generating new code.
  • Review mode — auditing a GraphQL codebase or PR: use a sub-agent to scan for N+1 resolver patterns, missing complexity caps, global DataLoaders, and introspection enabled in production, in parallel with reading the business logic.

Community default. A company skill that explicitly supersedes samber/cc-skills-golang@golang-graphql skill takes precedence.

Go GraphQL Best Practices

Both major libraries are schema-first: write SDL (.graphql files), bind Go resolvers. Choose based on project size and team preferences.

This skill is not exhaustive. Refer to each library's official documentation and code examples for current API signatures. Context7 can help as a discoverability platform.

Library Choice

Library Approach Type safety Build step Best for
github.com/99designs/gqlgen Codegen Compile-time go generate Large schemas, federation, strict types
github.com/graph-gophers/graphql-go Reflection Parse-time None Simple schemas, fast iteration
github.com/graphql-go/graphql Code-first Runtime None Avoid — verbose, no SDL

Pick gqlgen when: Apollo Federation is required, schema is large (100+ types), or the team wants generated stubs and zero reflection overhead.

Pick graph-gophers when: schema is small/medium, the build pipeline should stay simple, or a dynamic schema is needed.

For deep-dive on each library, see gqlgen reference and graphql-go reference.

Schema Design

# ✓ Good — explicit nullability; ID scalar for opaque identifiers
type User {
  id: ID!
  email: String! # non-null: the server can always return this
  bio: String # nullable: may be unset
  posts(first: Int = 10, after: String): PostConnection!
}

# ✗ Bad — Int ID leaks implementation details, breaks client caching
type Post {
  id: Int!
}

Nullability rule: mark a field ! only when the server can always return a value. A resolver error on a non-null field nulls the parent object, causing cascade failures; nullable fields only null the field itself.

Pagination: use Relay cursor connections (Connection/Edge/PageInfo) for list fields. Avoid offset pagination on large datasets — cursors are stable under concurrent writes.

Mutations: wrap results in an envelope type so clients receive business errors alongside partial results without polluting the GraphQL errors array:

type CreateUserPayload {
  user: User
  errors: [UserError!]!
}

Resolver Patterns

Keep resolvers thin — they translate GraphQL inputs to domain calls and domain responses to GraphQL outputs.

// ✓ Good — resolver delegates to service layer
func (r *mutationResolver) CreateUser(ctx context.Context, input model.CreateUserInput) (*model.CreateUserPayload, error) {
    user, err := r.userService.Create(ctx, input.Email, input.Name)
    if err != nil {
        return nil, formatError(err)
    }
    return &model.CreateUserPayload{User: toGQLUser(user)}, nil
}

// ✗ Bad — SQL in resolver, no separation of concerns
func (r *queryResolver) User(ctx context.Context, id string) (*model.User, error) {
    row := r.db.QueryRowContext(ctx, "SELECT * FROM users WHERE id = $1", id)
    // ...
}

Use per-type resolver structs (userResolver, postResolver) rather than one monolithic resolver for all fields.

N+1 Prevention (DataLoaders)

Each User.posts resolver fires a SQL query per user without batching — O(n) DB calls for n users. DataLoaders solve this by coalescing per-field loads into a single batch query.

Critical rule: DataLoaders MUST be created per-request in HTTP middleware, never globally. A global DataLoader caches across requests — stale data, potential cross-user data leakage.

// ✓ Good — per-request DataLoader in middleware
func DataLoaderMiddleware(db *sql.DB, next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        loaders := &Loaders{
            PostsByUserID: newPostsByUserIDLoader(r.Context(), db),
        }
        ctx := context.WithValue(r.Context(), loadersKey, loaders)
        next.ServeHTTP(w, r.WithContext(ctx))
    })
}

// ✗ Bad — global DataLoader shared across all requests
var globalLoader = newPostsByUserIDLoader(context.Background(), db)

In gqlgen, mark batched fields with resolver: true in gqlgen.yml to force a dedicated resolver method. See gqlgen reference for full DataLoader wiring.

Authentication and Authorization

Two-layer model:

  1. HTTP middleware — extract and validate tokens, stash identity in context.Context.
  2. Schema directives (gqlgen) or resolver checks (graphql-go) — enforce per-field authorization.
// HTTP middleware layer (both libraries)
func AuthMiddleware(next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        token := r.Header.Get("Authorization")
        user, err := validateToken(token)
        if err != nil {
            http.Error(w, "Unauthorized", http.StatusUnauthorized)
            return
        }
        ctx := context.WithValue(r.Context(), userKey, user)
        next.ServeHTTP(w, r.WithContext(ctx))
    })
}

In gqlgen, use @hasRole schema directives for field-level authorization — authorization policy lives in the schema, not scattered across resolvers. See gqlgen reference.

Error Handling

Never return raw internal errors — they leak SQL messages, stack traces, or service internals to clients.

// gqlgen — custom ErrorPresenter strips internal details
srv.SetErrorPresenter(func(ctx context.Context, err error) *gqlerror.Error {
    var gqlErr *gqlerror.Error
    if errors.As(err, &gqlErr) {
        return gqlErr // already formatted
    }
    // log internal err here
    return gqlerror.Errorf("internal error") // safe client message
})

// Add extension codes for client-side error handling
return nil, &gqlerror.Error{
    Message: "user not found",
    Extensions: map[string]any{"code": "NOT_FOUND"},
}

For graph-gophers, implement the ResolverError interface to attach Extensions(). See graphql-go reference.

Use graphql.AddError(ctx, err) in gqlgen for non-fatal field errors where the resolver can still return partial data.

For error wrapping patterns, see the samber/cc-skills-golang@golang-error-handling skill.

Subscriptions

Subscriptions use long-lived WebSocket connections. The critical discipline: always respect context cancellation — a leaked goroutine per disconnected client exhausts resources silently.

// ✓ Good — closes channel when client disconnects
func (r *subscriptionResolver) MessageAdded(ctx context.Context, room string) (\x3C-chan *model.Message, error) {
    ch := make(chan *model.Message, 1)
    sub := r.pubsub.Subscribe(room) // subscribe once before the goroutine
    go func() {
        defer close(ch) // always close; signals iteration to stop
        for {
            select {
            case \x3C-ctx.Done():
                return // client disconnected
            case msg := \x3C-sub:
                ch \x3C- msg
            }
        }
    }()
    return ch, nil
}

// ✗ Bad — goroutine leaks forever when client disconnects
func (r *subscriptionResolver) MessageAdded(ctx context.Context, room string) (\x3C-chan *model.Message, error) {
    ch := make(chan *model.Message, 1)
    go func() {
        for msg := range r.pubsub.Subscribe(room) {
            ch \x3C- msg // blocks forever after client gone
        }
    }()
    return ch, nil
}

Performance and Safety

Production GraphQL servers require explicit limits. Without them, a single deeply nested query exhausts CPU and memory.

// gqlgen — wire these into every production handler
srv := handler.NewDefaultServer(es)
srv.Use(extension.FixedComplexityLimit(200)) // max cost per query

// Gate introspection — only in non-production environments
if os.Getenv("ENV") != "production" {
    srv.Use(extension.Introspection{})
}

For graph-gophers: graphql.MaxDepth(10) and graphql.MaxParallelism(10) options at ParseSchema time.

Query allow-listing: in production, consider persisted queries (gqlgen APQ extension) to reject arbitrary query strings.

Common Mistakes

Mistake Why it matters Fix
N+1 queries in child resolvers One SQL per parent row → O(n) DB calls Use per-request DataLoader
Global DataLoader Cross-request cache — stale data, data leaks Create DataLoader in request middleware
Editing models_gen.go directly Next go generate wipes hand edits Use autobind or models.\x3CT>.model in gqlgen.yml
Forgetting go generate after schema change Resolver interface mismatch at compile time Re-run go run github.com/99designs/gqlgen generate
int field in graph-gophers resolver Library requires int32 for Int scalar Use int32 (or float64 for Float)
Introspection enabled in production Exposes full schema to attackers Gate with ENV check
No complexity cap Deeply nested query → CPU/memory DoS extension.FixedComplexityLimit(N)
Leaking DB errors from resolvers Exposes SQL internals to clients Wrap in ErrorPresenter / ResolverError
Subscription goroutine leak Client disconnect → goroutine runs forever defer close(ch) + select ctx.Done()
Nullable field for always-required data Clients must null-check everywhere Mark ! in schema; return error from resolver

Deep Dives

  • gqlgen reference — codegen workflow, gqlgen.yml, DataLoaders, Federation v2, directives
  • graphql-go reference — reflection resolver model, type mapping, tracing
  • Testing — gqlgen client harness, gqltesting, httptest patterns

Cross-References

  • → See samber/cc-skills-golang@golang-context skill for context propagation in resolvers and subscriptions
  • → See samber/cc-skills-golang@golang-error-handling skill for error wrapping and sentinel patterns
  • → See samber/cc-skills-golang@golang-testing skill for table-driven and integration test patterns
  • → See samber/cc-skills-golang@golang-observability skill for tracing and metrics in resolvers
  • → See samber/cc-skills-golang@golang-security skill for input validation and injection prevention
  • → See samber/cc-skills-golang@golang-database skill for N+1 query patterns and DataLoader database batching

References

If you encounter a bug or unexpected behavior in gqlgen, open an issue at https://github.com/99designs/gqlgen/issues.

If you encounter a bug or unexpected behavior in graph-gophers/graphql-go, open an issue at https://github.com/graph-gophers/graphql-go/issues.

Usage Guidance
This skill appears safe for normal Go GraphQL development. Before installing, be aware that it can guide the agent to edit code, run Go/git/curl commands, consult external documentation, and use sub-agents to scan your repository. Review generated code, approve commands carefully, and pin external Go tooling versions where practical.
Capability Analysis
Type: OpenClaw Skill Name: golang-graphql Version: 0.0.2 The golang-graphql skill bundle is a comprehensive and well-structured set of instructions for building GraphQL APIs in Go using gqlgen and graph-gophers/graphql-go. It promotes security best practices, such as disabling introspection in production, implementing query complexity limits, and preventing data leakage through proper error handling and per-request DataLoaders. While it requests access to potentially risky tools like Bash(curl:*) and WebFetch, these are contextually relevant for GraphQL development tasks (e.g., fetching schemas or testing endpoints), and there is no evidence of malicious intent, data exfiltration, or unauthorized execution in the code or instructions (SKILL.md, references/*.md).
Capability Assessment
Purpose & Capability
The skill purpose is consistent with its content: it provides Go GraphQL implementation guidance for gqlgen and graph-gophers/graphql-go. It can edit files and run Go-related commands, which is expected for a coding skill.
Instruction Scope
The instructions include use of background/sub-agents for code scanning, but the stated scope is limited to finding resolver patterns, N+1 issues, complexity caps, and similar GraphQL review tasks.
Install Mechanism
There is no install spec and no included executable code. Reference docs include user-directed `go run` setup examples for gqlgen, which are normal for this ecosystem but should be version-pinned in real projects.
Credentials
Allowed tools include Read/Edit/Write, Go, golangci-lint, git, curl, WebFetch, Agent, and Context7 documentation lookups. These are proportionate for a coding-assistant skill, but users should review generated code and command usage.
Persistence & Privilege
The artifacts do not request credentials, persistent background services, privileged system access, hidden storage, or ongoing autonomous behavior.
How to Use
  1. Make sure OpenClaw is installed (local or Docker)
  2. Run the install command in chat: /install golang-graphql
  3. After installation, invoke the skill by name or use /golang-graphql
  4. Provide required inputs per the skill's parameter spec and get structured output
Version History
v0.0.2
- Expanded documentation on best practices for schema design, resolver patterns, N+1 prevention, authentication, error handling, and subscriptions for GraphQL in Golang. - Added comparison table of major Go GraphQL libraries (`gqlgen`, `graphql-go`, and `graphql-go/graphql`). - Provided guidance on DataLoader usage, including safe per-request instantiation. - Clarified nullability rules and mutation patterns in schema definitions. - Outlined recommended project/library choices and integration strategies with examples. - Updated compatibility, persona, and usage instructions for coding agent environments.
Metadata
Slug golang-graphql
Version 0.0.2
License MIT-0
All-time Installs 0
Active Installs 0
Total Versions 1
Frequently Asked Questions

What is Golang Graphql?

Implements GraphQL APIs in Golang using gqlgen or graphql-go. Apply when building GraphQL servers, designing schemas, writing resolvers, handling subscriptio... It is an AI Agent Skill for Claude Code / OpenClaw, with 54 downloads so far.

How do I install Golang Graphql?

Run "/install golang-graphql" in the OpenClaw or Claude Code chat to install it in one step — no extra setup required.

Is Golang Graphql free?

Yes, Golang Graphql is completely free, licensed under MIT-0. You can download, install and use it at no cost.

Which platforms does Golang Graphql support?

Golang Graphql is cross-platform and runs anywhere OpenClaw / Claude Code is available (cross-platform).

Who created Golang Graphql?

It is built and maintained by Samuel Berthe (@samber); the current version is v0.0.2.

💬 Comments