Skip to content

Commit

Permalink
[feat] add create sails generator (#99)
Browse files Browse the repository at this point in the history
* feat(create-sails-generator): add basic generators

* feat(create-sails-generator): add page generator

* fix: remove unused util
  • Loading branch information
DominusKelvin authored Apr 10, 2024
1 parent 15cf781 commit b926c90
Show file tree
Hide file tree
Showing 20 changed files with 529 additions and 0 deletions.
21 changes: 21 additions & 0 deletions create-sails-generator/LICENSE
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
MIT License

Copyright (c) 2023 The Sailscasts Company

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
41 changes: 41 additions & 0 deletions create-sails-generator/README
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
# create-sails-generator: A Sails Generator for The Boring JavaScript Stack

This generator is specifically designed for The Boring JavaScript Stack, providing a plethora of commands for code generation and more to expedite your development process with The Boring JavaScript Stack.

## Features

- **Code Generation**: Quickly scaffold boilerplate code for pages, actions, and more.
- **Enhanced CLI Commands**: Extensive CLI commands tailored for Sails.js development.
- **Saves Time and Effort**: Automate repetitive tasks, saving you valuable development time.

## Installation

To use Create-Sails-Generator, you'll first need to have Node.js and npm installed on your machine. Then, follow these simple steps:

1. Install Create-Sails-Generator globally via npm:

```sh
npm i create-sails-generator --save-dev
```

2. Once installed, you can access the generator commands using the `sails generate` command.

## Usage

### Generating Files

You can generate various files using the generator with simple commands. For instance, to create a new page, use:

```sh
sails generate page dashboard
```

[Check out the docs](https://docs.sailscasts.com/boring-stack/generator) for more information on usage.

## License

`create-sails-generator` is licensed under the [MIT License](LICENSE.md).

## About

Create-Sails-Generator is maintained by The Sailscasts Company. It is part of [The Boring JavaScript Stack](https://docs.sailscasts.com/boring-stack) - The reliable full-stack JavaScript stack.
24 changes: 24 additions & 0 deletions create-sails-generator/generators/bad-request/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
const path = require('path')
/**
* Generates responses/badRequest.js file
*/
module.exports = {
before: function (scope, done) {
if (scope.force !== false) {
scope.force = true
}
scope.relPath = 'badRequest.js'
return done()
},
after: function (scope, done) {
console.log()
console.log(`Successfully generated ${scope.relPath}`)
console.log(' •-', `api/responses/${scope.relPath}`)
console.log()
return done()
},
targets: {
'./api/responses/:relPath': { copy: 'badRequest.js' }
},
templatesDirectory: path.resolve(__dirname, './templates')
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
/**
* badRequest.js
*
* A custom response.
*
* Example usage:
* ```
* return res.badRequest();
* // -or-
* return res.badRequest(optionalData);
* ```
*
* Or with actions2:
* ```
* exits: {
* somethingHappened: {
* responseType: 'badRequest'
* }
* }
* ```
*
* ```
* throw 'somethingHappened';
* // -or-
* throw { somethingHappened: optionalData }
* ```
*/

module.exports = function badRequest(optionalData) {
// Get access to `req` and `res`
const req = this.req
const res = this.res

// Define the status code to send in the response.
const statusCodeToSet = 400

// Check if it's an Inertia request
if (req.header('X-Inertia')) {
if (optionalData && optionalData.problems) {
const errors = {}
optionalData.problems.forEach((problem) => {
if (typeof problem === 'object') {
Object.keys(problem).forEach((propertyName) => {
const sanitizedProblem = problem[propertyName].replace(/\.$/, '') // Trim trailing dot
if (!errors[propertyName]) {
errors[propertyName] = [sanitizedProblem]
} else {
errors[propertyName].push(sanitizedProblem)
}
})
} else {
const regex = /"(.*?)"/
const matches = problem.match(regex)

if (matches && matches.length > 1) {
const propertyName = matches[1]
const sanitizedProblem = problem
.replace(/"([^"]+)"/, '$1')
.replace('\n', '')
.replace('·', '')
.trim()
if (!errors[propertyName]) {
errors[propertyName] = [sanitizedProblem]
} else {
errors[propertyName].push(sanitizedProblem)
}
}
}
})
req.session.errors = errors
return res.redirect(303, 'back')
}
}

// If not an Inertia request, perform the normal badRequest response
if (optionalData === undefined) {
sails.log.info('Ran custom response: res.badRequest()')
return res.sendStatus(statusCodeToSet)
} else if (_.isError(optionalData)) {
sails.log.info(
'Custom response `res.badRequest()` called with an Error:',
optionalData
)

if (!_.isFunction(optionalData.toJSON)) {
if (process.env.NODE_ENV === 'production') {
return res.sendStatus(statusCodeToSet)
} else {
return res.status(statusCodeToSet).send(optionalData.stack)
}
}
} else {
return res.status(statusCodeToSet).send(optionalData)
}
}
24 changes: 24 additions & 0 deletions create-sails-generator/generators/inertia-redirect/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
const path = require('path')
/**
* Generates responses/inertia.js file
*/
module.exports = {
before: function (scope, done) {
if (scope.force !== false) {
scope.force = true
}
scope.relPath = 'inertiaRedirect.js'
return done()
},
after: function (scope, done) {
console.log()
console.log(`Successfully generated ${scope.relPath}`)
console.log(' •-', `api/responses/${scope.relPath}`)
console.log()
return done()
},
targets: {
'./api/responses/:relPath': { copy: 'inertiaRedirect.js' }
},
templatesDirectory: path.resolve(__dirname, './templates')
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
// @ts-nocheck

const inertiaHeaders = {
INERTIA: 'X-Inertia',
LOCATION: 'X-Inertia-Location'
}

module.exports = function inertiaRedirect(url) {
const req = this.req
const res = this.res

if (req.get(inertiaHeaders.INERTIA)) {
res.set(inertiaHeaders.LOCATION, url)
}

return res.redirect(
['PUT', 'PATCH', 'DELETE'].includes(req.method) ? 303 : 409,
url
)
}
24 changes: 24 additions & 0 deletions create-sails-generator/generators/inertia/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
const path = require('path')
/**
* Generates responses/inertia.js file
*/
module.exports = {
before: function (scope, done) {
if (scope.force !== false) {
scope.force = true
}
scope.relPath = 'inertia.js'
return done()
},
after: function (scope, done) {
console.log()
console.log(`Successfully generated ${scope.relPath}`)
console.log(' •-', `api/responses/${scope.relPath}`)
console.log()
return done()
},
targets: {
'./api/responses/:relPath': { copy: 'inertia.js' }
},
templatesDirectory: path.resolve(__dirname, './templates')
}
72 changes: 72 additions & 0 deletions create-sails-generator/generators/inertia/templates/inertia.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
// @ts-nocheck
const { encode } = require('querystring')
module.exports = function inertia(data) {
const req = this.req
const res = this.res
const sails = req._sails

const sharedProps = sails.inertia.sharedProps
const sharedViewData = sails.inertia.sharedViewData
const rootView = sails.config.inertia.rootView

const allProps = {
...sharedProps,
...data.props
}

const allViewData = {
...sharedViewData,
...data.viewData
}

let url = req.url || req.originalUrl
const assetVersion = sails.config.inertia.version
const currentVersion =
typeof assetVersion === 'function' ? assetVersion() : assetVersion

const page = {
component: data.page,
version: currentVersion,
props: allProps,
url
}

// Implements inertia partial reload. See https://inertiajs.com/partial-reload
if (
req.get(inertiaHeaders.PARTIAL_DATA) &&
req.get(inertiaHeaders.PARTIAL_COMPONENT) === page.component
) {
const only = req.get(inertiaHeaders.PARTIAL_DATA).split(',')
page.props = only.length ? getPartialData(data.props, only) : page.props
}

const queryParams = req.query
if (req.method === 'GET' && Object.keys(queryParams).length) {
// Keep original request query params
url += `?${encode(queryParams)}`
}

if (req.get(inertiaHeaders.INERTIA)) {
res.set(inertiaHeaders.INERTIA, true)
res.set('Vary', 'Accept')
return res.json(page)
} else {
// Implements full page reload
return res.view(rootView, {
page,
viewData: allViewData
})
}
}

function getPartialData(props, only = []) {
return Object.assign({}, ...only.map((key) => ({ [key]: props[key] })))
}

const inertiaHeaders = {
INERTIA: 'X-Inertia',
VERSION: 'X-Inertia-Version',
PARTIAL_DATA: 'X-Inertia-Partial-Data',
PARTIAL_COMPONENT: 'X-Inertia-Partial-Component',
LOCATION: 'X-Inertia-Location'
}
70 changes: 70 additions & 0 deletions create-sails-generator/generators/page/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
const path = require('path')
const getUiFramework = require('../../utils/get-ui-framework')
const getComponentName = require('../../utils/get-component-name')
const getFileExtensionForUi = require('../../utils/get-file-extension-for-ui')
const getActionName = require('../../utils/get-action-name')
/**
* Generates responses/inertia.js file
*/

module.exports = {
before: function (scope, done) {
const appPackageJSON = require(`${scope.topLvlRootPath}/package.json`)
const uiFramework = getUiFramework(appPackageJSON)
let roughName

if (scope.name) {
roughName = scope.name
}

if (scope.args[0] && typeof scope.args[0] == 'string') {
roughName = scope.args[0]
}

if (!roughName) {
return done(
new Error(
'Missing argument: Please provide a name for the new page.\n' +
'(e.g. `profile` or `sign-up`).'
)
)
}

// Replace backslashes with proper slashes.
// (This is crucial for Windows compatibility.)
roughName = roughName.replace(/\\/g, '/')

scope.pageRelPath = roughName.replace(/\.+/g, '/')
scope.pagePath = scope.pageRelPath
console.log(scope.pagePath)
scope.pageRelPath += getFileExtensionForUi(uiFramework)
scope.uiFramework = uiFramework
if (uiFramework == 'react') {
scope.componentName = getComponentName(roughName)
}

scope.actionRelPath = getActionName(roughName)
scope.actionRelPath += '.js'
scope.actionFriendlyName = `View ${scope.pagePath}`
scope.actionDescription = `Display ${scope.pagePath} page`

return done()
},
after: function (scope, done) {
console.log()

console.log(`Successfully generated ${scope.pageRelPath}`)
console.log(' •-', `assets/js/pages/${scope.pageRelPath}`)

console.log(`Successfully generated ${scope.actionRelPath}`)
console.log(' •-', `api/controllers/${scope.actionRelPath}`)

console.log()
return done()
},
targets: {
'./assets/js/pages/:pageRelPath': { template: 'page.template' },
'./api/controllers/:actionRelPath': { template: 'action.template' }
},
templatesDirectory: path.resolve(__dirname, './templates')
}
Loading

0 comments on commit b926c90

Please sign in to comment.