@brainfish-ai/devdoc 0.1.21
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +33 -0
- package/README.md +415 -0
- package/bin/devdoc.js +13 -0
- package/dist/cli/commands/build.d.ts +5 -0
- package/dist/cli/commands/build.js +87 -0
- package/dist/cli/commands/check.d.ts +1 -0
- package/dist/cli/commands/check.js +143 -0
- package/dist/cli/commands/create.d.ts +24 -0
- package/dist/cli/commands/create.js +387 -0
- package/dist/cli/commands/deploy.d.ts +9 -0
- package/dist/cli/commands/deploy.js +433 -0
- package/dist/cli/commands/dev.d.ts +6 -0
- package/dist/cli/commands/dev.js +139 -0
- package/dist/cli/commands/init.d.ts +11 -0
- package/dist/cli/commands/init.js +238 -0
- package/dist/cli/commands/keys.d.ts +12 -0
- package/dist/cli/commands/keys.js +165 -0
- package/dist/cli/commands/start.d.ts +5 -0
- package/dist/cli/commands/start.js +56 -0
- package/dist/cli/commands/upload.d.ts +13 -0
- package/dist/cli/commands/upload.js +238 -0
- package/dist/cli/commands/whoami.d.ts +8 -0
- package/dist/cli/commands/whoami.js +91 -0
- package/dist/cli/index.d.ts +1 -0
- package/dist/cli/index.js +106 -0
- package/dist/config/index.d.ts +80 -0
- package/dist/config/index.js +133 -0
- package/dist/constants.d.ts +9 -0
- package/dist/constants.js +13 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.js +12 -0
- package/dist/utils/logger.d.ts +16 -0
- package/dist/utils/logger.js +61 -0
- package/dist/utils/paths.d.ts +16 -0
- package/dist/utils/paths.js +50 -0
- package/package.json +51 -0
- package/renderer/app/api/assets/[...path]/route.ts +123 -0
- package/renderer/app/api/assets/route.ts +124 -0
- package/renderer/app/api/assets/upload/route.ts +177 -0
- package/renderer/app/api/auth-schemes/route.ts +77 -0
- package/renderer/app/api/chat/route.ts +858 -0
- package/renderer/app/api/codegen/route.ts +72 -0
- package/renderer/app/api/collections/route.ts +1016 -0
- package/renderer/app/api/debug/route.ts +53 -0
- package/renderer/app/api/deploy/route.ts +234 -0
- package/renderer/app/api/device/route.ts +42 -0
- package/renderer/app/api/docs/route.ts +187 -0
- package/renderer/app/api/keys/regenerate/route.ts +80 -0
- package/renderer/app/api/openapi-spec/route.ts +151 -0
- package/renderer/app/api/projects/[slug]/route.ts +153 -0
- package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
- package/renderer/app/api/projects/register/route.ts +152 -0
- package/renderer/app/api/proxy/route.ts +149 -0
- package/renderer/app/api/proxy-stream/route.ts +168 -0
- package/renderer/app/api/redirects/route.ts +47 -0
- package/renderer/app/api/schema/route.ts +65 -0
- package/renderer/app/api/subdomains/check/route.ts +172 -0
- package/renderer/app/api/suggestions/route.ts +144 -0
- package/renderer/app/favicon.ico +0 -0
- package/renderer/app/globals.css +1103 -0
- package/renderer/app/layout.tsx +47 -0
- package/renderer/app/llms-full.txt/route.ts +346 -0
- package/renderer/app/llms.txt/route.ts +279 -0
- package/renderer/app/page.tsx +14 -0
- package/renderer/app/robots.txt/route.ts +84 -0
- package/renderer/app/sitemap.xml/route.ts +199 -0
- package/renderer/components/docs/index.ts +12 -0
- package/renderer/components/docs/mdx/accordion.tsx +169 -0
- package/renderer/components/docs/mdx/badge.tsx +132 -0
- package/renderer/components/docs/mdx/callouts.tsx +154 -0
- package/renderer/components/docs/mdx/cards.tsx +213 -0
- package/renderer/components/docs/mdx/changelog.tsx +120 -0
- package/renderer/components/docs/mdx/code-block.tsx +186 -0
- package/renderer/components/docs/mdx/code-group.tsx +421 -0
- package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
- package/renderer/components/docs/mdx/frame.tsx +112 -0
- package/renderer/components/docs/mdx/highlight.tsx +151 -0
- package/renderer/components/docs/mdx/iframe.tsx +134 -0
- package/renderer/components/docs/mdx/image.tsx +235 -0
- package/renderer/components/docs/mdx/index.ts +204 -0
- package/renderer/components/docs/mdx/mermaid.tsx +240 -0
- package/renderer/components/docs/mdx/param-field.tsx +200 -0
- package/renderer/components/docs/mdx/steps.tsx +113 -0
- package/renderer/components/docs/mdx/tabs.tsx +86 -0
- package/renderer/components/docs/mdx-renderer.tsx +100 -0
- package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
- package/renderer/components/docs/navigation/index.ts +8 -0
- package/renderer/components/docs/navigation/page-nav.tsx +64 -0
- package/renderer/components/docs/navigation/sidebar.tsx +515 -0
- package/renderer/components/docs/navigation/toc.tsx +113 -0
- package/renderer/components/docs/notice.tsx +105 -0
- package/renderer/components/docs-header.tsx +274 -0
- package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
- package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
- package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
- package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
- package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
- package/renderer/components/docs-viewer/agent/index.tsx +6 -0
- package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
- package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
- package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
- package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
- package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
- package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
- package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
- package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
- package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
- package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
- package/renderer/components/docs-viewer/content/index.tsx +29 -0
- package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
- package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
- package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
- package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
- package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
- package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
- package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
- package/renderer/components/docs-viewer/index.tsx +1466 -0
- package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
- package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
- package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
- package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
- package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
- package/renderer/components/docs-viewer/playground/index.tsx +671 -0
- package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
- package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
- package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
- package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
- package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
- package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
- package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
- package/renderer/components/docs-viewer/search/index.ts +2 -0
- package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
- package/renderer/components/docs-viewer/search/use-search.ts +117 -0
- package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
- package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
- package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
- package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
- package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
- package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
- package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
- package/renderer/components/theme-provider.tsx +11 -0
- package/renderer/components/theme-toggle.tsx +76 -0
- package/renderer/components/ui/badge.tsx +46 -0
- package/renderer/components/ui/button.tsx +59 -0
- package/renderer/components/ui/dialog.tsx +118 -0
- package/renderer/components/ui/dropdown-menu.tsx +257 -0
- package/renderer/components/ui/input.tsx +21 -0
- package/renderer/components/ui/label.tsx +24 -0
- package/renderer/components/ui/navigation-menu.tsx +168 -0
- package/renderer/components/ui/select.tsx +190 -0
- package/renderer/components/ui/spinner.tsx +114 -0
- package/renderer/components/ui/tabs.tsx +66 -0
- package/renderer/components/ui/tooltip.tsx +61 -0
- package/renderer/hooks/use-code-copy.ts +88 -0
- package/renderer/hooks/use-openapi-title.ts +44 -0
- package/renderer/lib/api-docs/agent/index.ts +6 -0
- package/renderer/lib/api-docs/agent/indexer.ts +323 -0
- package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
- package/renderer/lib/api-docs/agent/types.ts +116 -0
- package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
- package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
- package/renderer/lib/api-docs/auth/crypto.ts +89 -0
- package/renderer/lib/api-docs/auth/index.ts +4 -0
- package/renderer/lib/api-docs/code-editor/db.ts +164 -0
- package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
- package/renderer/lib/api-docs/code-editor/index.ts +6 -0
- package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
- package/renderer/lib/api-docs/code-editor/types.ts +105 -0
- package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
- package/renderer/lib/api-docs/codegen/har.ts +251 -0
- package/renderer/lib/api-docs/codegen/index.ts +159 -0
- package/renderer/lib/api-docs/factories.ts +151 -0
- package/renderer/lib/api-docs/index.ts +17 -0
- package/renderer/lib/api-docs/mobile-context.tsx +112 -0
- package/renderer/lib/api-docs/navigation-context.tsx +88 -0
- package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
- package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
- package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
- package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
- package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
- package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
- package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
- package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
- package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
- package/renderer/lib/api-docs/playground/context.tsx +107 -0
- package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
- package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
- package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
- package/renderer/lib/api-docs/playground/types.ts +35 -0
- package/renderer/lib/api-docs/types.ts +269 -0
- package/renderer/lib/api-docs/utils.ts +311 -0
- package/renderer/lib/cache.ts +193 -0
- package/renderer/lib/docs/config/index.ts +29 -0
- package/renderer/lib/docs/config/loader.ts +142 -0
- package/renderer/lib/docs/config/schema.ts +298 -0
- package/renderer/lib/docs/index.ts +12 -0
- package/renderer/lib/docs/mdx/compiler.ts +176 -0
- package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
- package/renderer/lib/docs/mdx/index.ts +26 -0
- package/renderer/lib/docs/navigation/generator.ts +348 -0
- package/renderer/lib/docs/navigation/index.ts +12 -0
- package/renderer/lib/docs/navigation/types.ts +123 -0
- package/renderer/lib/docs-navigation-context.tsx +80 -0
- package/renderer/lib/multi-tenant/context.ts +105 -0
- package/renderer/lib/storage/blob.ts +845 -0
- package/renderer/lib/utils.ts +6 -0
- package/renderer/next.config.ts +76 -0
- package/renderer/package.json +66 -0
- package/renderer/postcss.config.mjs +5 -0
- package/renderer/public/assets/images/screenshot.png +0 -0
- package/renderer/public/assets/logo/dark.svg +9 -0
- package/renderer/public/assets/logo/light.svg +9 -0
- package/renderer/public/assets/logo.svg +9 -0
- package/renderer/public/file.svg +1 -0
- package/renderer/public/globe.svg +1 -0
- package/renderer/public/icon.png +0 -0
- package/renderer/public/logo.svg +9 -0
- package/renderer/public/window.svg +1 -0
- package/renderer/tsconfig.json +28 -0
- package/templates/basic/README.md +139 -0
- package/templates/basic/assets/favicon.svg +4 -0
- package/templates/basic/assets/logo.svg +9 -0
- package/templates/basic/docs.json +47 -0
- package/templates/basic/guides/configuration.mdx +149 -0
- package/templates/basic/guides/overview.mdx +96 -0
- package/templates/basic/index.mdx +39 -0
- package/templates/basic/package.json +14 -0
- package/templates/basic/quickstart.mdx +92 -0
- package/templates/basic/vercel.json +6 -0
- package/templates/graphql/README.md +139 -0
- package/templates/graphql/api-reference/schema.graphql +305 -0
- package/templates/graphql/assets/favicon.svg +4 -0
- package/templates/graphql/assets/logo.svg +9 -0
- package/templates/graphql/docs.json +54 -0
- package/templates/graphql/guides/configuration.mdx +149 -0
- package/templates/graphql/guides/overview.mdx +96 -0
- package/templates/graphql/index.mdx +39 -0
- package/templates/graphql/package.json +14 -0
- package/templates/graphql/quickstart.mdx +92 -0
- package/templates/graphql/vercel.json +6 -0
- package/templates/openapi/README.md +139 -0
- package/templates/openapi/api-reference/openapi.json +419 -0
- package/templates/openapi/assets/favicon.svg +4 -0
- package/templates/openapi/assets/logo.svg +9 -0
- package/templates/openapi/docs.json +61 -0
- package/templates/openapi/guides/configuration.mdx +149 -0
- package/templates/openapi/guides/overview.mdx +96 -0
- package/templates/openapi/index.mdx +39 -0
- package/templates/openapi/package.json +14 -0
- package/templates/openapi/quickstart.mdx +92 -0
- package/templates/openapi/vercel.json +6 -0
|
@@ -0,0 +1,491 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* GraphQL Schema Parser
|
|
3
|
+
*
|
|
4
|
+
* Parses GraphQL SDL (Schema Definition Language) files into our internal types.
|
|
5
|
+
* Uses the graphql-js library for parsing.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import {
|
|
9
|
+
buildSchema,
|
|
10
|
+
type GraphQLSchema as GQLSchema,
|
|
11
|
+
type GraphQLObjectType,
|
|
12
|
+
type GraphQLInputObjectType,
|
|
13
|
+
type GraphQLEnumType,
|
|
14
|
+
type GraphQLInterfaceType,
|
|
15
|
+
type GraphQLUnionType,
|
|
16
|
+
type GraphQLScalarType as GQLScalarType,
|
|
17
|
+
type GraphQLField,
|
|
18
|
+
type GraphQLArgument as GQLArgument,
|
|
19
|
+
type GraphQLInputField as GQLInputField,
|
|
20
|
+
type GraphQLType,
|
|
21
|
+
type GraphQLOutputType,
|
|
22
|
+
type GraphQLInputType,
|
|
23
|
+
isObjectType,
|
|
24
|
+
isInputObjectType,
|
|
25
|
+
isEnumType,
|
|
26
|
+
isInterfaceType,
|
|
27
|
+
isUnionType,
|
|
28
|
+
isScalarType,
|
|
29
|
+
isNonNullType,
|
|
30
|
+
isListType,
|
|
31
|
+
isNamedType,
|
|
32
|
+
getNamedType,
|
|
33
|
+
} from 'graphql'
|
|
34
|
+
|
|
35
|
+
import {
|
|
36
|
+
formatTypeRef,
|
|
37
|
+
type GraphQLSchema,
|
|
38
|
+
type GraphQLSchemaInfo,
|
|
39
|
+
type GraphQLTypeDef,
|
|
40
|
+
type GraphQLTypeRef,
|
|
41
|
+
type GraphQLField as BFGraphQLField,
|
|
42
|
+
type GraphQLArgument as BFGraphQLArgument,
|
|
43
|
+
type GraphQLInputField as BFGraphQLInputField,
|
|
44
|
+
type GraphQLOperation,
|
|
45
|
+
type GraphQLDirective,
|
|
46
|
+
type GraphQLOperationType,
|
|
47
|
+
} from './types'
|
|
48
|
+
|
|
49
|
+
// ============================================================================
|
|
50
|
+
// Type Conversion Helpers
|
|
51
|
+
// ============================================================================
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Convert a GraphQL type to our type reference format
|
|
55
|
+
*/
|
|
56
|
+
function convertTypeToRef(type: GraphQLType | GraphQLOutputType | GraphQLInputType): GraphQLTypeRef {
|
|
57
|
+
if (isNonNullType(type)) {
|
|
58
|
+
return {
|
|
59
|
+
kind: 'NON_NULL',
|
|
60
|
+
name: null,
|
|
61
|
+
ofType: convertTypeToRef(type.ofType),
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
if (isListType(type)) {
|
|
66
|
+
return {
|
|
67
|
+
kind: 'LIST',
|
|
68
|
+
name: null,
|
|
69
|
+
ofType: convertTypeToRef(type.ofType),
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
if (isNamedType(type)) {
|
|
74
|
+
const namedType = getNamedType(type)
|
|
75
|
+
let kind: GraphQLTypeRef['kind'] = 'SCALAR'
|
|
76
|
+
|
|
77
|
+
if (isObjectType(namedType)) kind = 'OBJECT'
|
|
78
|
+
else if (isInputObjectType(namedType)) kind = 'INPUT_OBJECT'
|
|
79
|
+
else if (isEnumType(namedType)) kind = 'ENUM'
|
|
80
|
+
else if (isInterfaceType(namedType)) kind = 'INTERFACE'
|
|
81
|
+
else if (isUnionType(namedType)) kind = 'UNION'
|
|
82
|
+
else if (isScalarType(namedType)) kind = 'SCALAR'
|
|
83
|
+
|
|
84
|
+
return {
|
|
85
|
+
kind,
|
|
86
|
+
name: namedType.name,
|
|
87
|
+
ofType: null,
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
return {
|
|
92
|
+
kind: 'SCALAR',
|
|
93
|
+
name: 'Unknown',
|
|
94
|
+
ofType: null,
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* Convert GraphQL argument to our format
|
|
100
|
+
*/
|
|
101
|
+
function convertArgument(arg: GQLArgument): BFGraphQLArgument {
|
|
102
|
+
return {
|
|
103
|
+
name: arg.name,
|
|
104
|
+
description: arg.description || null,
|
|
105
|
+
type: convertTypeToRef(arg.type),
|
|
106
|
+
defaultValue: arg.defaultValue !== undefined ? JSON.stringify(arg.defaultValue) : null,
|
|
107
|
+
isRequired: isNonNullType(arg.type),
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Convert GraphQL input field to our format
|
|
113
|
+
*/
|
|
114
|
+
function convertInputField(field: GQLInputField): BFGraphQLInputField {
|
|
115
|
+
return {
|
|
116
|
+
name: field.name,
|
|
117
|
+
description: field.description || null,
|
|
118
|
+
type: convertTypeToRef(field.type),
|
|
119
|
+
defaultValue: field.defaultValue !== undefined ? JSON.stringify(field.defaultValue) : null,
|
|
120
|
+
isRequired: isNonNullType(field.type),
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
/**
|
|
125
|
+
* Convert GraphQL field to our format
|
|
126
|
+
*/
|
|
127
|
+
function convertField(field: GraphQLField<unknown, unknown>): BFGraphQLField {
|
|
128
|
+
return {
|
|
129
|
+
name: field.name,
|
|
130
|
+
description: field.description || null,
|
|
131
|
+
args: field.args.map(convertArgument),
|
|
132
|
+
type: convertTypeToRef(field.type),
|
|
133
|
+
isDeprecated: field.deprecationReason != null,
|
|
134
|
+
deprecationReason: field.deprecationReason || null,
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
// ============================================================================
|
|
139
|
+
// Type Definition Converters
|
|
140
|
+
// ============================================================================
|
|
141
|
+
|
|
142
|
+
/**
|
|
143
|
+
* Convert an object type to our format
|
|
144
|
+
*/
|
|
145
|
+
function convertObjectType(type: GraphQLObjectType): GraphQLTypeDef {
|
|
146
|
+
const fields = type.getFields()
|
|
147
|
+
return {
|
|
148
|
+
kind: 'OBJECT',
|
|
149
|
+
name: type.name,
|
|
150
|
+
description: type.description || null,
|
|
151
|
+
fields: Object.values(fields).map(convertField),
|
|
152
|
+
interfaces: type.getInterfaces().map(i => i.name),
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
/**
|
|
157
|
+
* Convert an input object type to our format
|
|
158
|
+
*/
|
|
159
|
+
function convertInputObjectType(type: GraphQLInputObjectType): GraphQLTypeDef {
|
|
160
|
+
const fields = type.getFields()
|
|
161
|
+
return {
|
|
162
|
+
kind: 'INPUT_OBJECT',
|
|
163
|
+
name: type.name,
|
|
164
|
+
description: type.description || null,
|
|
165
|
+
inputFields: Object.values(fields).map(convertInputField),
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/**
|
|
170
|
+
* Convert an enum type to our format
|
|
171
|
+
*/
|
|
172
|
+
function convertEnumType(type: GraphQLEnumType): GraphQLTypeDef {
|
|
173
|
+
return {
|
|
174
|
+
kind: 'ENUM',
|
|
175
|
+
name: type.name,
|
|
176
|
+
description: type.description || null,
|
|
177
|
+
enumValues: type.getValues().map(v => ({
|
|
178
|
+
name: v.name,
|
|
179
|
+
description: v.description || null,
|
|
180
|
+
isDeprecated: v.deprecationReason != null,
|
|
181
|
+
deprecationReason: v.deprecationReason || null,
|
|
182
|
+
})),
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
/**
|
|
187
|
+
* Convert an interface type to our format
|
|
188
|
+
*/
|
|
189
|
+
function convertInterfaceType(type: GraphQLInterfaceType, schema: GQLSchema): GraphQLTypeDef {
|
|
190
|
+
const fields = type.getFields()
|
|
191
|
+
const implementations = schema.getPossibleTypes(type)
|
|
192
|
+
|
|
193
|
+
return {
|
|
194
|
+
kind: 'INTERFACE',
|
|
195
|
+
name: type.name,
|
|
196
|
+
description: type.description || null,
|
|
197
|
+
fields: Object.values(fields).map(convertField),
|
|
198
|
+
possibleTypes: implementations.map(t => t.name),
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
/**
|
|
203
|
+
* Convert a union type to our format
|
|
204
|
+
*/
|
|
205
|
+
function convertUnionType(type: GraphQLUnionType): GraphQLTypeDef {
|
|
206
|
+
return {
|
|
207
|
+
kind: 'UNION',
|
|
208
|
+
name: type.name,
|
|
209
|
+
description: type.description || null,
|
|
210
|
+
possibleTypes: type.getTypes().map(t => t.name),
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
/**
|
|
215
|
+
* Convert a scalar type to our format
|
|
216
|
+
*/
|
|
217
|
+
function convertScalarType(type: GQLScalarType): GraphQLTypeDef {
|
|
218
|
+
return {
|
|
219
|
+
kind: 'SCALAR',
|
|
220
|
+
name: type.name,
|
|
221
|
+
description: type.description || null,
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
// ============================================================================
|
|
226
|
+
// Operation Extractors
|
|
227
|
+
// ============================================================================
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* Extract operations from a root type (Query, Mutation, Subscription)
|
|
231
|
+
*/
|
|
232
|
+
function extractOperations(
|
|
233
|
+
rootType: GraphQLObjectType | null | undefined,
|
|
234
|
+
operationType: GraphQLOperationType
|
|
235
|
+
): GraphQLOperation[] {
|
|
236
|
+
if (!rootType) return []
|
|
237
|
+
|
|
238
|
+
const fields = rootType.getFields()
|
|
239
|
+
|
|
240
|
+
return Object.values(fields).map(field => ({
|
|
241
|
+
id: `${operationType}-${field.name}`,
|
|
242
|
+
name: field.name,
|
|
243
|
+
description: field.description || null,
|
|
244
|
+
operationType,
|
|
245
|
+
args: field.args.map(convertArgument),
|
|
246
|
+
returnType: convertTypeToRef(field.type),
|
|
247
|
+
isDeprecated: field.deprecationReason != null,
|
|
248
|
+
deprecationReason: field.deprecationReason || null,
|
|
249
|
+
}))
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
/**
|
|
253
|
+
* Extract directives from schema
|
|
254
|
+
*/
|
|
255
|
+
function extractDirectives(schema: GQLSchema): GraphQLDirective[] {
|
|
256
|
+
return schema.getDirectives().map(directive => ({
|
|
257
|
+
name: directive.name,
|
|
258
|
+
description: directive.description || null,
|
|
259
|
+
locations: directive.locations as string[],
|
|
260
|
+
args: directive.args.map(convertArgument),
|
|
261
|
+
}))
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
// ============================================================================
|
|
265
|
+
// Main Parser Functions
|
|
266
|
+
// ============================================================================
|
|
267
|
+
|
|
268
|
+
/**
|
|
269
|
+
* Parse a GraphQL SDL string into our schema format
|
|
270
|
+
*/
|
|
271
|
+
export function parseGraphQLSchema(sdl: string): GraphQLSchema {
|
|
272
|
+
// Build the schema from SDL
|
|
273
|
+
const schema = buildSchema(sdl)
|
|
274
|
+
|
|
275
|
+
return convertGraphQLSchema(schema)
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
/**
|
|
279
|
+
* Convert a graphql-js schema object to our format
|
|
280
|
+
*/
|
|
281
|
+
export function convertGraphQLSchema(schema: GQLSchema): GraphQLSchema {
|
|
282
|
+
const typeMap = schema.getTypeMap()
|
|
283
|
+
const types: Record<string, GraphQLTypeDef> = {}
|
|
284
|
+
|
|
285
|
+
// Convert all types (excluding built-in types that start with __)
|
|
286
|
+
for (const [name, type] of Object.entries(typeMap)) {
|
|
287
|
+
if (name.startsWith('__')) continue // Skip introspection types
|
|
288
|
+
|
|
289
|
+
if (isObjectType(type)) {
|
|
290
|
+
types[name] = convertObjectType(type)
|
|
291
|
+
} else if (isInputObjectType(type)) {
|
|
292
|
+
types[name] = convertInputObjectType(type)
|
|
293
|
+
} else if (isEnumType(type)) {
|
|
294
|
+
types[name] = convertEnumType(type)
|
|
295
|
+
} else if (isInterfaceType(type)) {
|
|
296
|
+
types[name] = convertInterfaceType(type, schema)
|
|
297
|
+
} else if (isUnionType(type)) {
|
|
298
|
+
types[name] = convertUnionType(type)
|
|
299
|
+
} else if (isScalarType(type)) {
|
|
300
|
+
types[name] = convertScalarType(type)
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
// Get root types
|
|
305
|
+
const queryType = schema.getQueryType()
|
|
306
|
+
const mutationType = schema.getMutationType()
|
|
307
|
+
const subscriptionType = schema.getSubscriptionType()
|
|
308
|
+
|
|
309
|
+
// Schema info
|
|
310
|
+
const info: GraphQLSchemaInfo = {
|
|
311
|
+
description: schema.description || null,
|
|
312
|
+
queryType: queryType?.name || null,
|
|
313
|
+
mutationType: mutationType?.name || null,
|
|
314
|
+
subscriptionType: subscriptionType?.name || null,
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
return {
|
|
318
|
+
info,
|
|
319
|
+
types,
|
|
320
|
+
directives: extractDirectives(schema),
|
|
321
|
+
queries: extractOperations(queryType, 'query'),
|
|
322
|
+
mutations: extractOperations(mutationType, 'mutation'),
|
|
323
|
+
subscriptions: extractOperations(subscriptionType, 'subscription'),
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
|
|
327
|
+
/**
|
|
328
|
+
* Parse GraphQL SDL from a file content (handles both .graphql and .gql)
|
|
329
|
+
*/
|
|
330
|
+
export async function parseGraphQLFile(content: string): Promise<GraphQLSchema> {
|
|
331
|
+
return parseGraphQLSchema(content)
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
/**
|
|
335
|
+
* Validate that content is valid GraphQL SDL
|
|
336
|
+
*/
|
|
337
|
+
export function validateGraphQLSchema(sdl: string): { valid: boolean; error?: string } {
|
|
338
|
+
try {
|
|
339
|
+
buildSchema(sdl)
|
|
340
|
+
return { valid: true }
|
|
341
|
+
} catch (error) {
|
|
342
|
+
return {
|
|
343
|
+
valid: false,
|
|
344
|
+
error: error instanceof Error ? error.message : 'Invalid GraphQL schema',
|
|
345
|
+
}
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
/**
|
|
350
|
+
* Parse introspection query result to our schema format
|
|
351
|
+
*/
|
|
352
|
+
export function parseIntrospectionResult(introspectionResult: unknown): GraphQLSchema {
|
|
353
|
+
// The introspection result contains __schema with types, directives, etc.
|
|
354
|
+
// We need to convert this to our format
|
|
355
|
+
// For now, this is a placeholder - full implementation would parse the introspection JSON
|
|
356
|
+
|
|
357
|
+
const result = introspectionResult as {
|
|
358
|
+
__schema?: {
|
|
359
|
+
types?: Array<{
|
|
360
|
+
name: string
|
|
361
|
+
kind: string
|
|
362
|
+
description?: string
|
|
363
|
+
fields?: Array<{
|
|
364
|
+
name: string
|
|
365
|
+
description?: string
|
|
366
|
+
args?: unknown[]
|
|
367
|
+
type?: unknown
|
|
368
|
+
isDeprecated?: boolean
|
|
369
|
+
deprecationReason?: string
|
|
370
|
+
}>
|
|
371
|
+
inputFields?: unknown[]
|
|
372
|
+
enumValues?: unknown[]
|
|
373
|
+
interfaces?: unknown[]
|
|
374
|
+
possibleTypes?: unknown[]
|
|
375
|
+
}>
|
|
376
|
+
queryType?: { name: string }
|
|
377
|
+
mutationType?: { name: string } | null
|
|
378
|
+
subscriptionType?: { name: string } | null
|
|
379
|
+
directives?: unknown[]
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
if (!result.__schema) {
|
|
384
|
+
throw new Error('Invalid introspection result: missing __schema')
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
// This is a simplified implementation
|
|
388
|
+
// A full implementation would parse all the introspection data
|
|
389
|
+
const schema: GraphQLSchema = {
|
|
390
|
+
info: {
|
|
391
|
+
description: null,
|
|
392
|
+
queryType: result.__schema.queryType?.name || null,
|
|
393
|
+
mutationType: result.__schema.mutationType?.name || null,
|
|
394
|
+
subscriptionType: result.__schema.subscriptionType?.name || null,
|
|
395
|
+
},
|
|
396
|
+
types: {},
|
|
397
|
+
directives: [],
|
|
398
|
+
queries: [],
|
|
399
|
+
mutations: [],
|
|
400
|
+
subscriptions: [],
|
|
401
|
+
}
|
|
402
|
+
|
|
403
|
+
return schema
|
|
404
|
+
}
|
|
405
|
+
|
|
406
|
+
// ============================================================================
|
|
407
|
+
// Query Generation Helpers
|
|
408
|
+
// ============================================================================
|
|
409
|
+
|
|
410
|
+
/**
|
|
411
|
+
* Generate a sample query string for an operation
|
|
412
|
+
*/
|
|
413
|
+
export function generateOperationQuery(operation: GraphQLOperation): string {
|
|
414
|
+
const operationKeyword = operation.operationType
|
|
415
|
+
const args = operation.args
|
|
416
|
+
.map(arg => `$${arg.name}: ${formatTypeRefForQuery(arg.type)}`)
|
|
417
|
+
.join(', ')
|
|
418
|
+
|
|
419
|
+
const fieldArgs = operation.args
|
|
420
|
+
.map(arg => `${arg.name}: $${arg.name}`)
|
|
421
|
+
.join(', ')
|
|
422
|
+
|
|
423
|
+
const argsStr = args ? `(${args})` : ''
|
|
424
|
+
const fieldArgsStr = fieldArgs ? `(${fieldArgs})` : ''
|
|
425
|
+
|
|
426
|
+
return `${operationKeyword} ${operation.name}${argsStr} {
|
|
427
|
+
${operation.name}${fieldArgsStr} {
|
|
428
|
+
# Add fields here
|
|
429
|
+
}
|
|
430
|
+
}`
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
/**
|
|
434
|
+
* Format type ref for query variables
|
|
435
|
+
*/
|
|
436
|
+
function formatTypeRefForQuery(typeRef: GraphQLTypeRef): string {
|
|
437
|
+
if (typeRef.kind === 'NON_NULL' && typeRef.ofType) {
|
|
438
|
+
return `${formatTypeRefForQuery(typeRef.ofType)}!`
|
|
439
|
+
}
|
|
440
|
+
if (typeRef.kind === 'LIST' && typeRef.ofType) {
|
|
441
|
+
return `[${formatTypeRefForQuery(typeRef.ofType)}]`
|
|
442
|
+
}
|
|
443
|
+
return typeRef.name || 'Unknown'
|
|
444
|
+
}
|
|
445
|
+
|
|
446
|
+
/**
|
|
447
|
+
* Generate example variables for an operation
|
|
448
|
+
*/
|
|
449
|
+
export function generateExampleVariables(operation: GraphQLOperation): Record<string, unknown> {
|
|
450
|
+
const variables: Record<string, unknown> = {}
|
|
451
|
+
|
|
452
|
+
for (const arg of operation.args) {
|
|
453
|
+
variables[arg.name] = generateExampleValue(arg.type)
|
|
454
|
+
}
|
|
455
|
+
|
|
456
|
+
return variables
|
|
457
|
+
}
|
|
458
|
+
|
|
459
|
+
/**
|
|
460
|
+
* Generate example value for a type
|
|
461
|
+
*/
|
|
462
|
+
function generateExampleValue(typeRef: GraphQLTypeRef): unknown {
|
|
463
|
+
// Unwrap non-null
|
|
464
|
+
if (typeRef.kind === 'NON_NULL' && typeRef.ofType) {
|
|
465
|
+
return generateExampleValue(typeRef.ofType)
|
|
466
|
+
}
|
|
467
|
+
|
|
468
|
+
// Handle list
|
|
469
|
+
if (typeRef.kind === 'LIST' && typeRef.ofType) {
|
|
470
|
+
return [generateExampleValue(typeRef.ofType)]
|
|
471
|
+
}
|
|
472
|
+
|
|
473
|
+
// Handle named types
|
|
474
|
+
const typeName = typeRef.name
|
|
475
|
+
|
|
476
|
+
switch (typeName) {
|
|
477
|
+
case 'String':
|
|
478
|
+
return 'example'
|
|
479
|
+
case 'Int':
|
|
480
|
+
return 42
|
|
481
|
+
case 'Float':
|
|
482
|
+
return 3.14
|
|
483
|
+
case 'Boolean':
|
|
484
|
+
return true
|
|
485
|
+
case 'ID':
|
|
486
|
+
return 'abc123'
|
|
487
|
+
default:
|
|
488
|
+
// For custom types, return an empty object as placeholder
|
|
489
|
+
return {}
|
|
490
|
+
}
|
|
491
|
+
}
|