Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .changeset/rotten-tips-cry.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'create-kitql': patch
---

init create kitql
3 changes: 3 additions & 0 deletions packages/create-kitql/.eslintrc.cjs
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
module.exports = {
extends: ['eslint-config-kitql'],
}
1 change: 1 addition & 0 deletions packages/create-kitql/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
test-*
1 change: 1 addition & 0 deletions packages/create-kitql/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
# create-kitql
18 changes: 18 additions & 0 deletions packages/create-kitql/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
# ⚡ create kitql

_Part of [KitQL](https://github.com/jycouet/kitql#kitql), a set of tools helping 🫵 building efficient apps in a fast way._

## 📖 Read the doc

[![](https://img.shields.io/badge/Documentation%20of-create%20kitql-FF3E00.svg?style=flat&logo=&logoColor=E10098)](https://kitql.dev/docs/tools/06_vite-plugin-kit-routes)

## 🧑‍💻 Try it

```bash
npm create kitql@latest demo-kitql
```

## ⭐️ Join us

[![GitHub Repo stars](https://img.shields.io/github/stars/jycouet/kitql?logo=github&label=KitQL&color=FF3E00)](https://github.com/jycouet/kitql)

199 changes: 199 additions & 0 deletions packages/create-kitql/bin.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,199 @@
#!/usr/bin/env node
import * as p from '@clack/prompts'
import { bold, cyan, gray, green, italic } from '@kitql/helpers'
import { program, Option, InvalidArgumentError } from 'commander'
import fs, { readFileSync } from 'node:fs'
import path from 'node:path'
import { exit } from 'node:process'
import { fileURLToPath } from 'node:url'

// the first argument is the name of the project
let projectDir = process.argv[2]
let projectName = projectDir

const { version } = JSON.parse(fs.readFileSync(new URL('package.json', import.meta.url), 'utf-8'))
// eslint-disable-next-line no-console
console.log(`${gray(`create-kitql version ${version}`)}\n`)

// prepare options
const templatesDir = sourcePath(`./templates`)
const options = fs.readdirSync(templatesDir).map(templateDir => {
// in .meta.json you can find:
/** @type {{label?: string, hint?: string, apiUrl?: string}} */
let data = {}
const metaPath = path.join(templatesDir, templateDir, '.meta.json')
if (fs.existsSync(metaPath)) {
data = JSON.parse(readFileSync(metaPath, 'utf-8'))
}
return { ...data, value: templateDir }
})

program.argument('[project_name]', 'optional project name')
program.addOption(
new Option('-t, --template <template>', 'template you want to use').choices(
options.map(c => c.value),
),
)
program.addOption(
new Option('-s, --schema <schema>', '"local" or "http..."').argParser(value => {
if (value === 'local' || value.startsWith('http')) {
return value
}
throw new InvalidArgumentError('Should be "local" or "http..." or do not set it!')
}),
)

program.parse(process.argv)
const options_cli = program.opts()

p.intro('🎨 Welcome to Svelte UX!')

// if we weren't given a directory, then we should ask
if (!projectDir) {
const dir = await p.text({
message: `Where should we create your project?`,
placeholder: ' (press Enter to use the current directory)',
})

if (p.isCancel(dir)) {
process.exit(1)
}

if (dir) {
projectDir = dir
projectName = dir
} else {
projectDir = '.'
}
}

// if we were told to use the current directory then we need
// a more appropriate name for the project
if (projectDir === '.') {
projectName = 'kitql-demo'
}
let dirToCreate = true

// project location emtpy?
if (fs.existsSync(projectDir)) {
if (fs.readdirSync(projectDir).length > 0) {
const force = await p.confirm({
message:
'Target directory is not empty. Continue anyway? This might overwrite existing files.',
initialValue: false,
})
dirToCreate = false

// bail if `force` is `false` or the user cancelled with Ctrl-C
if (force !== true) {
process.exit(1)
}
}
}

// create the project directory if necessary
if (dirToCreate && !fs.existsSync(projectDir)) {
fs.mkdirSync(projectDir)
}

const template = options_cli.template
? options_cli.template
: await p.select({
message: 'Which template do you want to use?',
initialValue: 'starter',
options,
})
if (p.isCancel(template)) {
process.exit(1)
}
const templateDir = path.join(templatesDir, template)
const templateMeta = options.find(option => option.value === template)
if (!templateMeta) {
// this will never happen, but it helps to types later
exit(1)
}

copy(
sourcePath(path.join(templatesDir, template)),
projectDir,
{
PROJECT_NAME: projectName,
SVELTE_UX_VERSION: version,
},
{ '.meta.gitignore': '.gitignore' },
['.meta.json'],
)

p.outro(`🎉 Everything is ready!

👉 Next Steps
0️⃣ Go to your project : cd ${green(projectDir)}
1️⃣ Install dependencies : ${green(`npm i`)} | ${green(`pnpm i`)}
2️⃣ Start your application : ${green(`npm run dev`)} | ${green(`pnpm dev`)}`)

// eslint-disable-next-line no-console
console.log(
gray(
italic(
`${bold('❔ More help')} ` +
`at ${cyan('https://kitql.dev/')} ` +
`(📄 Docs, ⭐ Github, 📣 Discord, ...)\n`,
),
),
)

// Function to copy files recursively
function copy(
/** @type {string} */ sourceDir,
/** @type {string} */ destDir = projectDir,
/** @type {Record<string, string>} */ transformMap = {},
/** @type {Record<string, string>} */ transformFileMap = {},
/** @type {string[]} */ ignoreList = [],
) {
if (!fs.existsSync(destDir)) {
fs.mkdirSync(destDir)
}

const files = fs.readdirSync(sourceDir)
for (const fileSource of files) {
const fileDest = Object.entries(transformFileMap).reduce((acc, [key, value]) => {
return acc.replace(key, value)
}, fileSource)
// const file = fileSource.replace(".meta.gitignore", ".gitignore")
const sourceFilePath = path.join(sourceDir, fileSource)
const sourceRelative = path.relative(templateDir, sourceFilePath)
// skip the ignore list
if (!ignoreList.includes(sourceRelative)) {
const destFilePath = path.join(destDir, fileDest)

const stats = fs.statSync(sourceFilePath)

// files need to be copied and potentially transformed
if (stats.isFile()) {
// read the source file
const source = fs.readFileSync(sourceFilePath)

// apply any transformations if necessary
const transformed = Object.entries(transformMap).reduce((prev, [pattern, value]) => {
return prev.replaceAll(pattern, value)
}, source.toString())

// write the result
fs.writeFileSync(destFilePath, transformed)
}
// if we run into a directory then we should keep going
else if (stats.isDirectory()) {
copy(sourceFilePath, destFilePath, transformMap, transformFileMap, ignoreList)
}
}
}
}

function sourcePath(/** @type {string} */ path) {
return fileURLToPath(new URL(path, import.meta.url).href)
}

// function pCancel(cancelText = 'Operation cancelled.') {
// p.cancel(cancelText)
// process.exit(1)
// }
42 changes: 42 additions & 0 deletions packages/create-kitql/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
{
"name": "create-kitql",
"description": "A CLI for creating new projects... A lot of kind of projects!",
"keywords": [
"cli",
"svelte",
"sveltekit"
],
"version": "0.0.1",
"license": "MIT",
"type": "module",
"repository": {
"type": "git",
"url": "https://github.com/jycouet/kitql",
"directory": "packages/create-kitql",
"homepage": "https://github.com/jycouet/kitql/tree/main/packages/create-kitql"
},
"bin": "./bin.js",
"scripts": {
"dev": "node bin.js",
"lint": "kitql-lint",
"format": "kitql-lint --fix"
},
"devDependencies": {
"@types/node": "^20.9.2",
"eslint-config-kitql": "workspace:*"
},
"dependencies": {
"@clack/prompts": "^0.7.0",
"@kitql/helpers": "workspace:*",
"commander": "^11.1.0"
},
"sideEffects": false,
"publishConfig": {
"directory": "dist",
"access": "public"
},
"files": [
"templates",
"bin.js"
]
}
13 changes: 13 additions & 0 deletions packages/create-kitql/templates/kit-routes/.eslintignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
.DS_Store
node_modules
/build
/.svelte-kit
/package
.env
.env.*
!.env.example

# Ignore files for PNPM, NPM and YARN
pnpm-lock.yaml
package-lock.json
yarn.lock
31 changes: 31 additions & 0 deletions packages/create-kitql/templates/kit-routes/.eslintrc.cjs
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
/** @type { import("eslint").Linter.FlatConfig } */
module.exports = {
root: true,
extends: [
'eslint:recommended',
'plugin:@typescript-eslint/recommended',
'plugin:svelte/recommended',
'prettier'
],
parser: '@typescript-eslint/parser',
plugins: ['@typescript-eslint'],
parserOptions: {
sourceType: 'module',
ecmaVersion: 2020,
extraFileExtensions: ['.svelte']
},
env: {
browser: true,
es2017: true,
node: true
},
overrides: [
{
files: ['*.svelte'],
parser: 'svelte-eslint-parser',
parserOptions: {
parser: '@typescript-eslint/parser'
}
}
]
};
10 changes: 10 additions & 0 deletions packages/create-kitql/templates/kit-routes/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
.DS_Store
node_modules
/build
/.svelte-kit
/package
.env
.env.*
!.env.example
vite.config.js.timestamp-*
vite.config.ts.timestamp-*
9 changes: 9 additions & 0 deletions packages/create-kitql/templates/kit-routes/.meta..gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
.DS_Store
node_modules
/build
/.svelte-kit
/package
.env
!.env.example
vite.config.js.timestamp-*
vite.config.ts.timestamp-*
4 changes: 4 additions & 0 deletions packages/create-kitql/templates/kit-routes/.meta.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
{
"label": "Demo - Vite Kit Routes",
"hint": "Never go out of sync with your routes again! No more 🤞, now be confident ✅!"
}
1 change: 1 addition & 0 deletions packages/create-kitql/templates/kit-routes/.npmrc
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
engine-strict=true
13 changes: 13 additions & 0 deletions packages/create-kitql/templates/kit-routes/.prettierignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
.DS_Store
node_modules
/build
/.svelte-kit
/package
.env
.env.*
!.env.example

# Ignore files for PNPM, NPM and YARN
pnpm-lock.yaml
package-lock.json
yarn.lock
8 changes: 8 additions & 0 deletions packages/create-kitql/templates/kit-routes/.prettierrc
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
{
"useTabs": true,
"singleQuote": true,
"trailingComma": "none",
"printWidth": 100,
"plugins": ["prettier-plugin-svelte"],
"overrides": [{ "files": "*.svelte", "options": { "parser": "svelte" } }]
}
Loading