Wasp TypeScript config (*.wasp.ts)
This document assumes your app works with Wasp >= 0.15.
If you haven't migrated your app yet, follow the migration instructions and verify everything works. After that, come back here and try out the new Wasp TS config.
This feature is currently in early preview and we are actively working on it.
In Wasp, you normally define/configure the high level of your app (pages, routes, queries, actions, auth, ...) in a main.wasp
file in the root of your project. In main.wasp
you write in Wasp's DSL (domain-specific language), which is a simple configuration language similar to JSON but smarter.
Wasp 0.15 introduces the Wasp TS config, an alternative way to define the high level of your app via main.wasp.ts
! Although it looks similar to how you would do it in main.wasp
, the difference is that you write in TypeScript, not in Wasp's DSL.
Wasp TS config is an early preview feature, meaning it is a little rough and not yet where it could be, but it does work. We think it's pretty cool already, and you can try it out now. If you do, please share your feedback and ideas with us on our GitHub or Discord . This is crucial for us to be able to shape this feature in the best possible way!
Motivation
- Out-of-the-box support in all editors.
- Less maintenance on our side.
- More flexibility for you while writing the config.
- It will enable us to easily add support for multiple Wasp files in the future.
- A great foundation for the Full Stack Modules (FSM) that are a part of our future plans.
How to switch from the Wasp DSL config to the Wasp TS config
Go into the Wasp project you want to switch to the Wasp TS config (or create a new Wasp project if you want to try it out like that). Make sure you are on Wasp >= 0.15 and your project is working.
Rename
tsconfig.json
file totsconfig.src.json
and add"include": ["src"]
entry to the top level (next tocompilerOptions
, not in them!):tsconfig.src.json{
"compilerOptions": {
...
},
...
"include": ["src"]
}Create a new
tsconfig.json
file with the following content:tsconfig.json{
"files": [],
"references": [
{ "path": "./tsconfig.src.json" },
{ "path": "./tsconfig.wasp.json" }
]
}Create a new
tsconfig.wasp.json
file with the following content:tsconfig.wasp.json{
"compilerOptions": {
"skipLibCheck": true,
"target": "ES2022",
"isolatedModules": true,
"moduleDetection": "force",
// linting
"strict": true,
"noUnusedLocals": true,
"noUnusedParameters": true,
"module": "NodeNext",
"noEmit": true,
"lib": ["ES2023"],
},
"include": ["main.wasp.ts"]
}Add
"type": "module"
to the top level of yourpackage.json
, if you don't have it yet:package.json{
"type": "module",
...
}Rename the
main.wasp
file tomain.wasp.old
. You'll want to use it as a reference while writingmain.wasp.ts
.Run
wasp clean
andrm package-lock.json
. This ensures you start from a clean state.Run
wasp ts-setup
. This command will add thewasp-config
package to yourpackage.json
'sdevDependencies
.Create an empty
main.wasp.ts
file and rewrite yourmain.wasp.old
in it but in TypeScript.Check out the reference main.wasp.ts file below for details on what the TypeScript API for configuring Wasp looks like. In short, you'll have to:
- Import
App
fromwasp-config
- Create a new
app
object withnew App()
. - Use the
app
object to define parts of your web app likeauth
,pages
,query
,api
... - Export the
app
from your file using a default export.
You can manually do the rewrite using the reference file and TS types as guides (IDE support should work for you in
main.wasp.ts
), or you can (and we recommend it!) give the reference main.wasp.ts file to the LLM of your choice and tell it to rewrite yourmain.wasp
while following the format in the reference file: we had great results with this!- Import
Run
wasp start
to run your app! If you got everything right, your app should work exactly like it did before. The only difference is that it's now reading the Wasp config frommain.wasp.ts
instead ofmain.wasp
.tipDon't forget, during
wasp start
, to have the database running or do the db migrations if needed, as you would normally when running your app in development.That is it, you are now using Wasp TS config! You can delete
main.wasp.old
file now if you still have it around.
If you run wasp clean
or remove node_modules
on your own, you will have to rerun wasp ts-setup
!
Got stuck on any of these steps? Let us know in our Discord and we will help!
What next?
Experiment
Play with the Wasp TS config, get the feel of it, and see if you can find ways to improve it. Here are some ideas you can experiment with:
- How would you reduce the boilerplate in
main.wasp.ts
file? Helper functions, loops? - Can you imagine a better API or better abstractions? If you can, what would that look like? Perhaps you can even implement it on top of our API?
- Give a try at implementing your own file-based routing if that is what you like: you are now in Turing complete language and have access to the disk!
- Surprise us!
Feedback
Whatever you end up doing, we would love it if you would let us know how it was and show us what you did.
We do have some immediate ideas of our own about what we want to improve, but we want to hear what you thought of, what you liked or disliked, or what you came up with. Even if you just found it all good, or just a single thing you didn't or did like, that is also valuable feedback and we would love to hear it!
Let us know on our GitHub or, even better, in our Discord .
Reference main.wasp.ts file
import { App } from 'wasp-config'
const app = new App('todoApp', {
title: 'ToDo App',
wasp: { version: '^0.15.0' },
// head: []
});
app.webSocket({
fn: { import: 'webSocketFn', from: '@src/webSocket' },
// autoConnect: false
});
app.auth({
userEntity: 'User',
methods: {
discord: {
configFn: { import: 'config', from: '@src/auth/discord' },
userSignupFields: { import: 'userSignupFields', from: '@src/auth/discord' }
},
google: {
configFn: { import: 'config', from: '@src/auth/google' },
userSignupFields: { import: 'userSignupFields', from: '@src/auth/google' }
},
gitHub: {
configFn: { import: 'config', from: '@src/auth/github.js' },
userSignupFields: { import: 'userSignupFields', from: '@src/auth/github.js' }
},
// keycloak: {},
// email: {
// userSignupFields: { import: 'userSignupFields', from: '@src/auth/email' },
// fromField: {
// name: 'ToDO App',
// email: 'mihovil@ilakovac.com'
// },
// emailVerification: {
// getEmailContentFn: { import: 'getVerificationEmailContent', from: '@src/auth/email' },
// clientRoute: 'EmailVerificationRoute',
// },
// passwordReset: {
// getEmailContentFn: { import: 'getPasswordResetEmailContent', from: '@src/auth/email' },
// clientRoute: 'PasswordResetRoute'
// }
// },
},
onAuthFailedRedirectTo: '/login',
onAuthSucceededRedirectTo: '/profile',
onBeforeSignup: { import: 'onBeforeSignup', from: '@src/auth/hooks.js' },
onAfterSignup: { import: 'onAfterSignup', from: '@src/auth/hooks.js' },
onBeforeOAuthRedirect: { import: 'onBeforeOAuthRedirect', from: '@src/auth/hooks.js' },
onBeforeLogin: { import: 'onBeforeLogin', from: '@src/auth/hooks.js' },
onAfterLogin: { import: 'onAfterLogin', from: '@src/auth/hooks.js' }
});
app.server({
setupFn: { importDefault: 'setup', from: '@src/serverSetup' },
middlewareConfigFn: { import: 'serverMiddlewareFn', from: '@src/serverSetup' },
});
app.client({
rootComponent: { import: 'App', from: '@src/App' },
setupFn: { importDefault: 'setup', from: '@src/clientSetup' }
});
app.db({
seeds: [
{ import: 'devSeedSimple', from: '@src/dbSeeds' },
]
});
app.emailSender({
provider: 'SMTP',
defaultFrom: { email: 'test@test.com' }
});
const loginPage = app.page('LoginPage', {
component: { importDefault: 'Login', from: '@src/pages/auth/Login' }
});
app.route('LoginRoute', { path: '/login', to: loginPage });
app.query('getTasks', {
fn: { import: 'getTasks', from: '@src/queries' },
entities: ['Task']
});
app.action('createTask', {
fn: { import: 'createTask', from: '@src/actions' },
entities: ['Task']
});
app.apiNamespace('bar', {
middlewareConfigFn: { import: 'barNamespaceMiddlewareFn', from: '@src/apis' },
path: '/bar'
});
app.api('barBaz', {
fn: { import: 'barBaz', from: '@src/apis' },
auth: false,
entities: ['Task'],
httpRoute: ['GET', '/bar/baz']
});
app.job('mySpecialJob', {
executor: 'PgBoss',
perform: {
fn: { import: 'foo', from: '@src/jobs/bar' },
executorOptions: {
pgBoss: { retryLimit: 1 }
}
},
entities: ['Task']
});
export default app;