- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我想在我的 UserService 中注入(inject)存储库。
但我不知道该怎么做。
我正在使用 typescript、typedi 和 sequelize。
我认为,服务的加载速度比加载器快。
当我尝试注入(inject)我在数据库加载器中设置的存储库时,会发生错误。
像这样的错误: ServiceNotFoundError:在容器中找不到具有“存储库”标识符的服务。在使用前通过显式调用“Container.set”函数或使用“@Service()”装饰器注册它。
所以,我用 console.log 检查了“userRepo”,结果是未定义的。
我还检查了 CreateUser 的 Container.get('repositories'),它是集中加载的。我的意思是,我可以获得我的 Container 实例。
我只是无法在构造函数中加载存储库实例。
我应该怎么做才能在构造函数中加载存储库?
我应该将 sequelize 更改为 typeorm 来加载它吗?
// ** UserService.ts **
import { Inject, Service } from 'typedi';
import { UserRepository } from '../repositories/user.repository';
import { UserCreationAttributes } from '../models/interface/User.interface';
import { User, UserModel, UserStatic } from '../models/User';
@Service()
export default class UserService {
constructor(@Inject('repositories') private userRepo: UserRepository) {}
public async CreateUser(userData: UserCreationAttributes): Promise<boolean> {
try {
await this.userRepo.create(userData);
return true;
} catch (err) {
console.log(err);
return false;
}
}
}
// ** Database Loader **
import { Sequelize } from 'sequelize';
import config from '../config';
import Logger from './logger';
import { UserStatic } from '../models/User';
import { FeedStatic } from '../models/Feed';
import { CommentStatic } from '../models/Comment';
import { VerificationStatic } from '../models/Verification';
import { initializeModels } from '../models';
import { initializeRepositories, Repositories } from '../repositories';
import { Container } from 'typedi';
export interface Models {
User: UserStatic;
Feed: FeedStatic;
Comment: CommentStatic;
Verification: VerificationStatic;
}
export default async function loadSequelize() {
const sequelize = new Sequelize(
config.database,
config.databaseUsername,
config.databasePassword,
{
host: config.databaseHost,
port: config.databasePort,
dialect: 'postgres',
},
);
try {
await sequelize.authenticate();
const models: Models = initializeModels(sequelize);
const repositories: Repositories = initializeRepositories(models);
await sequelize.sync({ force: true });
// This part might be loaded after services were loaded
Container.set('models', models);
Container.set('repositories', repositories);
console.log('load finish');
} catch (err) {
Logger.error(err);
}
}
// ** ./repositories/index.ts **
import { Models } from '../loaders/database';
import { UserRepository } from './user.repository';
export interface Repositories {
UserRepository: UserRepository;
}
export const initializeRepositories = (models: Models): Repositories => {
const usersRepository = new UserRepository(models.User);
const repositories: Repositories = {
UserRepository: usersRepository,
};
return repositories;
};
// ** ./repositories/base.repository.ts **
import { Model, BuildOptions, FindOptions } from 'sequelize/types';
import { IFilter } from './filter/base.filter';
export type RichModel = typeof Model & {
new (values?: Record<string, unknown>, options?: BuildOptions): Model;
};
export interface IMeta {
globalCount: number;
countAfterFiltering: number;
}
export interface IWithMeta<M extends Model> {
meta: IMeta;
data: M[];
}
export abstract class BaseRepository<
M extends Model,
C extends object,
F extends IFilter = IFilter
> {
constructor(public _model: RichModel, private filterFactory: new () => F) {}
private async getCount(where?: Record<string, unknown>): Promise<number> {
const count = await this._model.count({ where });
return count;
}
async getAll(params?: FindOptions, filter?: F): Promise<IWithMeta<M>> {
const { from: offset, count: limit } = filter || {};
const result = await this._model.findAndCountAll({
order: [['id', 'ASC']],
offset: offset,
limit: limit,
...params,
});
const globalCount = await this.getCount();
const countAfterFiltering = ((result.count as unknown) as Record<
string,
unknown
>[]).length;
return {
meta: { globalCount, countAfterFiltering },
data: result.rows as M[],
};
}
async getById(id: string | number): Promise<M> {
const result = await this._model.findByPk(id);
return result as M;
}
async get(where: Record<string, unknown>): Promise<M> {
const result = await this._model.findOne({ where });
return result as M;
}
async updateById(id: string | number, data: C): Promise<M> {
const result = await this._model.update(data, {
where: { id },
returning: true,
});
const [, models] = result;
return models[0] as M;
}
async deleteById(id: string | number): Promise<void> {
await this._model.destroy({
where: { id },
});
}
async create(data: C): Promise<M> {
const model = await this._model.create(data);
return (model as unknown) as M;
}
}
// ** ./repositories/user.repository.ts
import { BaseRepository, IWithMeta, RichModel } from './base.repository';
import { UserModel, UserStatic } from '../models/User';
import { UserCreationAttributes } from '../models/interface/User.interface';
import { IFilter } from './filter/base.filter';
import { UserFilter } from './filter/user.filter';
import { Service } from 'typedi';
@Service()
export class UserRepository extends BaseRepository<
UserModel,
UserCreationAttributes,
IFilter
> {
constructor(private model: UserStatic) {
super(<RichModel>model, IFilter);
}
async getAllUsers(): Promise<IWithMeta<UserModel>> {
const users = await this.getAll();
return users;
}
async getOneByFilter({
email,
password,
}: UserFilter): Promise<UserModel | null> {
const user = await this.model.findOne({
where: {
email,
password,
},
});
return user;
}
async getAdminOneByFilter({
email,
password,
}: UserFilter): Promise<UserModel | null> {
const user = await this.model.findOne({
where: {
email,
password,
isAdmin: true,
},
});
return user;
}
}
最佳答案
首先,试着把你的initializeRepositories
在 reflect-metadata
之后立即调用电话调用,以测试执行顺序是否会影响。您不需要在数据库连接文件中调用。
根据我在旧文档中看到的,使用 Container.set
要设置所有存储库,您需要提供一个包含对象的数组,其中包含对象的 id。Container.set([{ id: 'userRepository', value: new UserRepository() }])
如果您只需要设置一个存储库:Container.set('userRepository', new UserRepository())
在上述两种情况下,您都可以使用:Container.get('userRepository')
;
如果您想将引用保留为 repositories
您的 initializeRepositories
方法是可以的,但是用法必须是这样的:
@Inject('repositories') repositories: { UserRepository: UserRepository};
repositories.UserRepository
如果您不需要通过该方法设置存储库,您可以放置一个装饰器
@Service('userRepository')
在您的
UserRepository
类,它会工作。
关于node.js - 我想使用 typedi 在 Service 构造函数中注入(inject) Repositories 容器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67309598/
谁能解释一下原因: (define a (lambda() (cons a #f))) (car (a)) ==> procedure ((car (a))) ==> (procedure . #f)
这是 PyBrain 网站的摘录。我了解大部分正在发生的事情,但是一行让我完全难住了。我以前从未在 python 代码中看到过这样的东西。这是整个循环,对于上下文: for c in [0,
我是gradle / groovy的新手。我想创建将做一些事情的自定义任务。我的第一个问题是任务完成时该如何做?我可以覆盖doFirst / doLast闭包吗?也许我可以重写某些在开始和结束时都会执
我刚刚开始评估 MS 企业库。他们使用以下指令来获取实例: var customerDb = EnterpriseLibraryContainer.Current.GetInstance("C
这是我的 if else Ansible 逻辑.. - name: Check certs exist stat: path=/etc/letsencrypt/live/{{ rootDomain
我正在使用construct 2.8 对一些失传已久的 Pascal 程序创建的一些文件的 header 进行逆向工程。 header 由许多不同的记录组成,其中一些是可选的,我不确定顺序是否固定。
我在将 getchar() 的输入放入 char *arr[] 数组时遇到问题。我这样做的原因是因为输入数据(将是一个带有命令行参数的文件)将存储在一个 char 指针数组中以传递给 execvp 函
通常我们不能约束类型参数 T派生自密封类型(例如 struct 类型)。这将毫无意义,因为只有一种类型适合,因此不需要泛型。所以约束如下: where T : string 或: where T :
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve th
#include using namespace std; class A { private: int m_i; friend int main(int argc, char cons
这个问题在这里已经有了答案: Are there legitimate uses for JavaScript's "with" statement? (33 个答案) 关闭 9 年前。 我有这个代
在this answer我看到了下一个 Bash 结构。 yes "$(< file.txt)" 什么意思 "$(< file.txt)" ? 我明白了 命令替换 - $(command)用命令的结
if (a == 1) //do something else if (a == 2) //do something else if (a == 3) //do somethi
关于构造的快速简单的问题。 我有以下用于将项目添加到 ListView 的代码。 ListViewItem item = new ListViewItem(); item.Text = file; i
我想使用 std::vector 来控制给定的内存。首先,我很确定这不是好的做法,但好奇心占了上风,无论如何我都想知道如何做到这一点。 我遇到的问题是这样的方法: vector getRow(unsi
下面显示了一段简单的javascript: var mystring = ("random","ignored","text","h") + ("ello world") 这个字符串会生成 hello
在 Java 中,创建对象的标准方法是使用 MyClass name = new MyClass(); 我也经常看到构造 new MyClass() { /*stuff goes in here*/
我正在编写 C++ ndarray 类。我需要动态大小和编译时大小已知的数组(分别分配自由存储和分配堆栈)。我想支持从嵌套的 std::initializer_list 进行初始化。 动态大小的没问题
我正在将一个项目从 Visual Studio 2005 转换为 Visual Studio 2008,并提出了上述结构。 using Castle.Core.Resource; using Cast
我想知道我在这里的想法是否正确,我主要针对接口(interface)进行编程,所以我想知道下面的类是否应该通过 DI 注入(inject),或者我应该自己实例化一个类... 注意:这些服务保存在我的核
我是一名优秀的程序员,十分优秀!