6

我正在尝试使用 NestJS 和 NATS 微服务。有很好的文档来设置基本的请求-响应。

我所做的是以下内容:

运行本地 NATS 服务器。

设置我main.ts连接到服务器:

async function bootstrap() {
  const app = await NestFactory.createMicroservice(AppModule, {
    options: {
      url: "nats://localhost:4222",
    },
    transport: Transport.NATS,
  });
  app.listen(() => console.log("Microservice is listening"));
}
bootstrap();

创建了一个 ClientProxyFactory 来发回消息:

export const NatsClientProvider: Provider = {
  inject: [ConfigService],
  provide: NatsClientProviderId,
  useFactory: async (config: ConfigService) =>
    ClientProxyFactory.create({
      options: {
        servers: config.getNatsConfig().servers,
      },
      transport: Transport.NATS,
    }),
};

设置控制器app.controller.ts以响应特定模式:

@Controller()
export class AppController {
  constructor(
    private readonly appService: AppService,
    @Inject(NatsClientProviderId) private readonly natsClient: ClientProxy,
  ) {}

  @MessagePattern("hello")
  async getHello(data: string) {
    console.log("data: ", data);
    console.log("getHello!!");
    await this.natsClient.send("hello", this.appService.getHello());
    return this.appService.getHello();
  }

  async onModuleInit() {
    await this.natsClient.connect();
    console.log("Nats connected!");
  }

设置测试文件以尝试发送请求-响应消息:

import { connect } from "ts-nats";

async function start() {
  const nc = await connect({
    servers: ["nats://localhost:4222"],
  });

  const msg = await nc.request("hello", 5000, "me");
  console.log("msg: ", msg);
}

start();

当我运行我的 Nest 应用程序时,我可以在 NATS 服务器日志中看到正确创建的订阅。

当我运行test.ts文件时,它会超时NatsError: Request timed out.。但是,我可以看到我的控制台日志(尽管数据是undefined即使我在发布的消息中指定它。

return和方法都client.send无法从应用程序接收消息。

任何帮助表示赞赏!

编辑:仍在研究并坚持这个问题。在微服务文档的“发送消息”部分,它说“模式必须等于在 @MessagePattern() 装饰器中定义的模式,而有效负载是我们想要传输到另一个微服务的消息。”。如果我这样做,Nest 应用程序会检测到它发送的消息并陷入无限循环,即发送消息并永远来回接收相同的消息。

4

2 回答 2

4

要避免控制器中的无限循环,请删除该natsClient.send语句。MessagePattern在您的情况下,将自动发送包含您从函数返回的数据的回复this.appService.getHello()

@MessagePattern("hello")
async getHello(data: string) {
  console.log("data: ", data);
  return "Hello World!";
}

Nest 要求您发送一个长id属性(任何字符串都可以),以便它能够回复消息。只需将其包含在数据 json 中:

// Nest expects the data to have the following structure
const reply = await nc.request("hello", 500, JSON.stringify({ data: "Hello", id: "myid" }));
console.log({ reply });

在您的嵌套日志中,您将看到以下日志条目:

data: Hello

在您的测试脚本中,您会看到:

{ reply:
   { subject: '_INBOX.GJGL6RJFYXKMCF8CWXO0HB.GJGL6RJFYXKMCF8CWXO0B5',
     sid: 1,
     reply: undefined,
     size: 50,
     data: '{"err":null,"response":"Hello World!","id":"myid"}' 
} }
于 2019-06-14T13:17:25.347 回答
2

使用 ClientProxy 时,send返回emitObservables。你需要“激活”那些让他们做任何事情。所以你可以要么subscribe给他们,要么把它改成 Promise。

因为你正在使用await你可能想要做

await this.natsClient.send("hello", this.appService.getHello()).toPromise();
于 2019-06-13T18:32:31.150 回答