lasthourcloud/CONTRIBUTING.md

9.9 KiB

👋 Welcome

First of all, thank you for considering contributing to my project! It means a lot 💜.

🙋 Want to help?

If you begin in GitHub contribution, you can find the first contribution and follow this guide.

Follow the introduction to get started then start contributing!

This is a little list of what you can do to help the project:

👋 Introduction

Setup with Github codespaces

If you have github codespaces enabled then you can just create a codespace and run pnpm dev to run your the dev environment. All the required dependencies and packages has been configured for you already.

Setup with Gitpod

If you have a Gitpod, you can just create a workspace from this repository, run pnpm install && pnpm db:push && pnpm db:seed and then pnpm dev. All the required dependencies and packages has been configured for you already.

Setup locally in your machine

🔴 At the moment, Coolify doesn't support Windows. You must use Linux or MacOS. Consider using Gitpod or Github Codespaces.

  • Fork the project
  • Clone your fork repo to local
  • Create a new branch
  • Push to your fork repo
  • Create a pull request: https://github.com/coollabsio/coolify/compare
  • Write a proper description
  • Open the pull request to review against next branch

🧑‍💻 Developer contribution

Technical skills required

  • Languages: Node.js / Javascript / Typescript
  • Framework JS/TS: SvelteKit & Fastify
  • Database ORM: Prisma.io
  • Docker Engine API

How to start after you set up your local fork?

Prerequisites

  1. Due to the lock file, this repository is best with pnpm. I recommend you try and use pnpm because it is cool and efficient!

  2. You need to have Docker Engine installed locally.

  3. You need to have Docker Compose Plugin installed locally.

  4. You need to have GIT LFS Support installed locally.

Optional:

  1. To test Heroku buildpacks, you need pack binary installed locally.

Steps for local setup

  1. Copy apps/api/.env.template to apps/api/.env.template and set the COOLIFY_APP_ID environment variable to something cool.

  2. Install dependencies with pnpm install.

  3. Need to create a local SQlite database with pnpm db:push.

    This will apply all migrations at db/dev.db.

  4. Seed the database with base entities with pnpm db:seed

  5. You can start coding after starting pnpm dev.


Database migrations

During development, if you change the database layout, you need to run pnpm db:push to migrate the database and create types for Prisma. You also need to restart the development process.

If the schema is finalized, you need to create a migration file with pnpm db:migrate <nameOfMigration> where nameOfMigration is given by you. Make it sense. :)


How to add new services

You can add any open-source and self-hostable software (service/application) to Coolify if the following statements are true:

  • Self-hostable (obviously)
  • Open-source
  • Maintained (I do not want to add software full of bugs)

Backend

There are 5 steps you should make on the backend side.

  1. Create Prisma / database schema for the new service.
  2. Add supported versions of the service.
  3. Update global functions.
  4. Create API endpoints.
  5. Define automatically generated variables.

I will use Umami as an example service.

Create Prisma / Database schema for the new service.

You only need to do this if you store passwords or any persistent configuration. Mostly it is required by all services, but there are some exceptions, like NocoDB.

Update Prisma schema in prisma/schema.prisma.

  • Add new model with the new service name.
  • Make a relationship with Service model.
  • In the Service model, the name of the new field should be with low-capital.
  • If the service needs a database, define a publicPort field to be able to make it's database public, example field name in case of PostgreSQL: postgresqlPublicPort. It should be a optional field.

If you are finished with the Prisma schema, you should update the database schema with pnpm db:push command.

You must restart the running development environment to be able to use the new model

If you use VSCode/TLS, you probably need to restart the Typescript Language Server to get the new types loaded in the running environment.

Add supported versions

Supported versions are hardcoded into Coolify (for now).

You need to update supportedServiceTypesAndVersions function at src/apps/api/src/lib/supportedVersions.ts. Example JSON:

     {
       // Name used to identify the service internally
       name: 'umami',
       // Fancier name to show to the user
       fancyName: 'Umami',
       // Docker base image for the service
       baseImage: 'ghcr.io/mikecao/umami',
       // Optional: If there is any dependent image, you should list it here
       images: [],
       // Usable tags
       versions: ['postgresql-latest'],
       // Which tag is the recommended
       recommendedVersion: 'postgresql-latest',
       // Application's default port, Umami listens on 3000
       ports: {
         main: 3000
       }
     }

Add required functions/properties

  1. Add the new service to the includeServices variable in apps/api/src/lib/services/common.ts, so it will be included in all places in the database queries where it is required.
const include: any = {
	destinationDocker: true,
	persistentStorage: true,
	serviceSecret: true,
	minio: true,
	plausibleAnalytics: true,
	vscodeserver: true,
	wordpress: true,
	ghost: true,
	meiliSearch: true,
	umami: true // This line!
};
  1. Update the database update query with the new service type to configureServiceType function in apps/api/src/lib/services/common.ts. This function defines the automatically generated variables (passwords, users, etc.) and it's encryption process (if applicable).
[...]
else if (type === 'umami') {
		const postgresqlUser = cuid();
		const postgresqlPassword = encrypt(generatePassword());
		const postgresqlDatabase = 'umami';
		const hashSalt = encrypt(generatePassword(64));
		await prisma.service.update({
			where: { id },
			data: {
				type,
				umami: {
					create: {
						postgresqlDatabase,
						postgresqlPassword,
						postgresqlUser,
						hashSalt,
					}
				}
			}
		});
	}
  1. Add field details to apps/api/src/lib/services/serviceFields.ts, so every component will know what to do with the values (decrypt/show it by default/readonly)
export const umami = [{
	name: 'postgresqlUser',
	isEditable: false,
	isLowerCase: false,
	isNumber: false,
	isBoolean: false,
	isEncrypted: false
}]
  1. Add service deletion query to removeService function in apps/api/src/lib/services/common.ts

  2. You need to add start process for the new service in apps/api/src/routes/api/v1/services/handlers.ts

See startUmamiService() function as example.

  1. You need to add a custom logo at apps/ui/src/lib/components/svg/services as a svelte component and export it in apps/ui/src/lib/components/svg/services/index.ts

    SVG is recommended, but you can use PNG as well. It should have the isAbsolute variable with the suitable CSS classes, primarily for sizing and positioning.

  2. You need to include it the logo at:

  1. By default the URL and the name frontend forms are included in apps/ui/src/routes/services/[id]/_Services/_Services.svelte.

    If you need to show more details on the frontend, such as users/passwords, you need to add Svelte component to apps/ui/src/routes/services/[id]/_Services with an underscore.

    For example, see other here.

Good job! 👏