Skip to content

Latest commit

 

History

History
324 lines (229 loc) · 8.48 KB

model.md

File metadata and controls

324 lines (229 loc) · 8.48 KB

Models

Cotton provides a type-safe way to work with your database using models. You can define a model by creating a plain class decorated with @Model.

import { Model, Column } from "https://deno.land/x/cotton/mod.ts";

@Model("users")
class User {
  @Primary()
  id!: number;

  @Column()
  email!: string;

  @Column()
  age!: number;

  @Column()
  createdAt!: Date;
}

The @Model decorator takes an optional argument for the table name of your model. By default, if the class name is User, it will look up for user table in your database.

A model must have a primary column. And at this point, Cotton only supports auto incremental primary keys.

Then, you define each columns by marking the class properties with @Column. Cotton is smart enough to determine the data type of that particular column using TypeScript types. However, you can still customize your column types by passing the type option like below.

@Column({ type: DataType.String })
email!: string;

There are still a plenty of room for customization. You can provide a default value of a column withdefault, define a custom name for your column using name, and much more.

@Column({ default: false })
isActive!: string;

@Column({ default: () => new Date() })
createdAt!: Date;

@Column({ name: "created_at" }) // different column name on the database
createdAt!: Date;

TypeScript configuration

Keep in mind that this feature requires a custom TypeScript configuration to tell Deno that we want to use TypeScript decorators, which currently still an experimental feature.

// tsconfig.json

{
  "compilerOptions": {
    "experimentalDecorators": true,
    "emitDecoratorMetadata": true
  }
}

Model manager

In order to perform queries within a model, you can use the model manager provided by the database connection.

const db = await connect({
  type: "sqlite",
  // other options...
});

const manager = db.getManager();

Once you get the manager instance, you can basically perform anything to that models. Let's start by creating a new user!

const user = new User();
user.email = "[email protected]";
user.age = 16;
user.createdAt = new Date();
await manager.save(user);

To update an existing model, you can perform save again. Cotton is smart enough to decide whether the model needs to be inserted or updated.

user.email = "[email protected]";
await manager.save(user);

You can also perform insert and update multiple models using save by passing an array as the argument.

const user1 = new User();
user1.email = "[email protected]";
await manager.save(user1);
user1.email = "[email protected]";

const user2 = new User();
user2.email = "[email protected]";

const post1 = new Post();
post1.title = "Spoon";

await manager.save([
  user1, // Since it's already saved, it will perform update.
  user2, // This record will inserted.
  post1, // You can also save a totally different model at once!
]);

Querying models

Everything you need to query a model is in the query method.

const users = await manager.query(User).all();

for (const user of users) {
  console.log(user); // User { email: 'a@b.com', age: 16, ... }
}

The query method returns an instance of ModelQuery which works like a query builder. You can also filter records by a certain conditions by chaining it with where, or, and not.

await manager.query(User).where("email", "[email protected]").all();

To fetch a single record, you can use first instead of all.

const user = await manager.query(User).where("email", "[email protected]").first();

console.log(user); // User { email: 'a@b.com', age: 16, ... }

Counting models

You can count how many models match given conditions using count method.

const count = await manager.query(User).where("isActive", true).count();

Updating models

To update multiple models at once, you can use the update method and pass the data. This will not return the updated models, use save instead.

await manager.query(User).where("isActive", true).update({ isActive: false });

Deleting models

Manager API has remove methods which allows you to remove a model you fetched from the database.

await manager.remove(user);

You can also remove multiple models by passing an array as the argument. This will remove all models in a single query.

await manager.remove([user1, user2, user3]);

To delete models that match given conditions, you can end your model query with delete method.

await manager.query(User).where("isActive", false).delete();

Relations

Cotton allows you to setup model relations with ease. Currently, Cotton only supports one-to-many relations.

@Model()
class User {
  @Primary()
  id!: number;

  @Column()
  email: string;

  @HasMany(() => User, "user_id")
  posts: Post[];
}

@Model()
class Post {
  @Primary()
  id!: number;

  @Column()
  title: string;

  @BelongsTo(() => User, "user_id")
  user: User;
}

As you can see, a user can have multiple posts, but the a post belongs to a single user.

Saving relations

Saving relations can be done using save method alone.

const post1 = new Post();
post1.title = "Post 1";
await manager.save(post1);

const post2 = new Post();
post2.title = "Post 2";
await manager.save(post2);

const user = new User();
user.email = "[email protected]";
user.posts = [post1, post2];
await manager.save(user);

You also can reverse it.

const user = new User();
user.email = "[email protected]";
await manager.save(user);

const post1 = new Post();
post1.title = "Post 1";
post1.user = user;
await manager.save(post1);

const post2 = new Post();
post2.title = "Post 1";
post2.user = user;
await manager.save(post2);

Fetching relations

By default, find and findOne doesn't fetch your relations. To fetch them, you need to explicitly say which relations you want to include.

const users = await manager.query(User).include("posts").all();
const post = await manager.query(Post).include("user").first();

Base model

If you find it difficult to use model manager, base model might be a perfect solution for you.

In a nutshell, base model is a model that extends the BaseModel class. This class provides you the exact same functionalities as the model manager, but you can call them directly from your model class.

@Model("users")
class User extends BaseModel {
  @Column()
  email!: string;

  @Column()
  age: number;

  @Column()
  createdAt!: Date;
}

The most important thing you need to do when working with base models is registering those models to your database connection. Not registering your models before executing queries with it can cause fatal errors.

const db = await connect({
  type: "sqlite",
  // other configs...
  models: [User],
});

Here's how to can perform query to a base model.

const user = await User.query().where("id", 1).first();

Inserting new item:

const user = new User();
user.email = "[email protected]";
user.age = 16;
user.createdAt = new Date();
await user.save();

// Alternatively...
const user = await User.insert({
  email: "[email protected]",
  age: 16,
  createdAt: new Date(),
});

Removing an item:

const user = await User.query().where("id", 1).first();
await user.remove();

What's the difference?

The most obvious difference between model manager and base model is that model manager is model agnostic. Which means it will work with any models you have.

In model manager, your model is just a plain class that only act as a representation of your table schema in the database. In order to do interact with the database, you need to use your model manager. Some people find it safer than the base models because it seperates the business logic and the schema definition. You can see this pattern a lot in Java frameworks such as Hibernate.

Base models on the other hand, your models act as both model manager and the schema. You can see this pattern in Laravel's Eloquent and ActiveRecord. A lot of people find this pattern easier to use because once you have access to the model class, you can basically do anything with it.

Which one should I use?

It's really up to you! I personally think it depends on where you came from. If you came from Java development and you already familiar with JPA, you probably want to use model manager. However, if you came from PHP, Ruby, or Node.js, base model probably looks more natural to you.