2024-03 spring-29-integrations-channels

This commit is contained in:
Vladimir Ivanov
2024-07-22 18:58:37 +03:00
parent 21c63e4b96
commit fe84b151b3
14 changed files with 617 additions and 0 deletions
@@ -0,0 +1,33 @@
HELP.md
target/
!.mvn/wrapper/maven-wrapper.jar
!**/src/main/**/target/
!**/src/test/**/target/
### STS ###
.apt_generated
.classpath
.factorypath
.project
.settings
.springBeans
.sts4-cache
### IntelliJ IDEA ###
.idea
*.iws
*.iml
*.ipr
### NetBeans ###
/nbproject/private/
/nbbuild/
/dist/
/nbdist/
/.nb-gradle/
build/
!**/src/main/**/build/
!**/src/test/**/build/
### VS Code ###
.vscode/
@@ -0,0 +1,47 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>ru.otus</groupId>
<artifactId>integrations-channels-exercise</artifactId>
<version>1.0-SNAPSHOT</version>
<parent>
<groupId>ru.otus</groupId>
<artifactId>integrations-channels</artifactId>
<version>1.0</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-integration</artifactId>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-messaging</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<scope>provided</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
@@ -0,0 +1,17 @@
package ru.otus.spring.integration;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.integration.annotation.IntegrationComponentScan;
import lombok.extern.slf4j.Slf4j;
@SpringBootApplication
@IntegrationComponentScan
@Slf4j
public class App {
public static void main(String[] args) {
SpringApplication.run(App.class, args);
}
}
@@ -0,0 +1,63 @@
package ru.otus.spring.integration;
import static java.util.Objects.isNull;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import org.springframework.boot.CommandLineRunner;
import org.springframework.integration.support.MessageBuilder;
import org.springframework.messaging.Message;
import org.springframework.messaging.PollableChannel;
import org.springframework.messaging.SubscribableChannel;
import org.springframework.stereotype.Component;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
@Slf4j
@Component
@RequiredArgsConstructor
public class AppRunner implements CommandLineRunner {
final PollableChannel queueChannel;
final SubscribableChannel subscribableDirectChannel;
@Override
public void run(String... args) throws Exception {
log.warn("INIT");
for (int i = 0; i < 10; i++) {
queueChannel.send(MessageBuilder.withPayload("Start " + i).build());
}
log.warn("INIT FINISH");
Thread.sleep(5000);
log.warn("START");
subscribableDirectChannel.subscribe((msg) -> log.warn("Receive msg: {}", msg));
ScheduledExecutorService executor = Executors.newSingleThreadScheduledExecutor();
executor.scheduleWithFixedDelay(() -> {
log.warn("I am here!!!");
Message<?> receivedMessage = queueChannel.receive(5000);
if (isNull(receivedMessage)) {
return;
}
subscribableDirectChannel.send(receivedMessage);
}, 100, 300, TimeUnit.MILLISECONDS);
log.warn("START FINISH");
log.warn("");
queueChannel.send(MessageBuilder.withPayload("Hello").build());
log.warn("");
queueChannel.send(MessageBuilder.withPayload("Hello2").build());
Thread.sleep(2_000);
log.warn("");
queueChannel.send(MessageBuilder.withPayload("Hello3").build());
Thread.sleep(3_000);
executor.shutdown();
}
}
@@ -0,0 +1,23 @@
package ru.otus.spring.integration;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.integration.channel.QueueChannel;
import org.springframework.integration.dsl.DirectChannelSpec;
import org.springframework.integration.dsl.MessageChannels;
import org.springframework.messaging.PollableChannel;
@Configuration
public class IntegrationConfig {
@Bean
public PollableChannel queueChannel() {
return new QueueChannel(100);
}
@Bean
public DirectChannelSpec subscribableDirectChannel() {
return MessageChannels.direct("subscribableDirectChannel");
}
}
@@ -0,0 +1,5 @@
logging:
level:
root: info
org.springframework.integration: debug
@@ -0,0 +1,98 @@
package ru.otus.spring.integration;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import java.util.Map;
import org.junit.jupiter.api.Test;
import org.springframework.messaging.Message;
import org.springframework.messaging.MessageHeaders;
import org.springframework.messaging.support.ErrorMessage;
import org.springframework.messaging.support.GenericMessage;
import org.springframework.messaging.support.MessageBuilder;
@SuppressWarnings("all")
public class MessagesTest {
@Test
public void testCreateSimpleGenericMessage() {
// TODO: Создайте сообщение с payload-ом "Hello" с помощью конструктора
Message<String> message = null;
assertNotNull(message);
assertEquals(GenericMessage.class, message.getClass());
assertNotNull(message.getPayload());
assertEquals("Hello", message.getPayload());
}
@Test
public void testCreateGenericMessage() {
// TODO: Создайте сообщение с пользователем с помощью конструктора
Message<User> message = null;
assertNotNull(message);
assertEquals(GenericMessage.class, message.getClass());
assertNotNull(message.getPayload());
assertEquals(new User("John", 23), message.getPayload());
}
@Test
public void testGenericMessageWithHeaders() {
// TODO: Создайте сообщение с payload-ом "Hello" и header-ом "to":"World"
Map<String, Object> headers = null;
Message<String> message = null;
assertNotNull(message);
assertEquals("Hello", message.getPayload());
assertEquals("World", message.getHeaders().get("to", String.class));
}
@Test
public void testGenericMessageWithMessageHeaders() {
// TODO: Создайте сообщение с payload-ом "Hello" и header-ом "to":"World"
MessageHeaders headers = null;
Message<String> message = null;
assertNotNull(message);
assertEquals("Hello", message.getPayload());
assertEquals("World", message.getHeaders().get("to", String.class));
}
@Test
public void testErrorMessage() {
// TODO: Создайте сообщение об ошибки с объектом NullPointerException внутри
Message errorMessage = null;
assertNotNull(errorMessage);
assertEquals(ErrorMessage.class, errorMessage.getClass());
assertEquals(NullPointerException.class, errorMessage.getPayload().getClass());
}
@Test
public void testMessageBuilder() {
// TODO: Создайте сообщение с payload-ом "Hello" и header-ом "to":"World" с помощью MessageBuilder
Message message = null;
assertNotNull(message);
assertEquals("Hello", message.getPayload());
assertEquals("World", message.getHeaders().get("to", String.class));
}
@Test
public void testBuildFromMessage() {
Message<User> original = MessageBuilder
.withPayload(new User("Kate", 30))
.setHeader("processor", "userService")
.build();
// TODO: Создайте новое сообщение с теми же payload и header-ами c помощью MessageBuilder
Message<User> newMessage = null;
assertNotNull(newMessage);
assertEquals(original.getPayload(), newMessage.getPayload());
assertEquals(original.getHeaders().get("processor"), newMessage.getHeaders().get("processor"));
}
}
@@ -0,0 +1,35 @@
package ru.otus.spring.integration;
import java.util.Objects;
public class User {
private final String name;
private final int age;
public User(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public int getAge() {
return age;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (!(o instanceof User user)) return false;
return age == user.age &&
Objects.equals(name, user.name);
}
@Override
public int hashCode() {
return Objects.hash(name, age);
}
}
@@ -0,0 +1,46 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>ru.otus</groupId>
<artifactId>integrations-channels-solution</artifactId>
<version>1.0-SNAPSHOT</version>
<parent>
<groupId>ru.otus</groupId>
<artifactId>integrations-channels</artifactId>
<version>1.0</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-integration</artifactId>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-messaging</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<scope>provided</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
@@ -0,0 +1,80 @@
package ru.otus.spring.integration;
import static java.util.Objects.isNull;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.integration.annotation.IntegrationComponentScan;
import org.springframework.integration.channel.QueueChannel;
import org.springframework.integration.dsl.DirectChannelSpec;
import org.springframework.integration.dsl.MessageChannels;
import org.springframework.integration.support.MessageBuilder;
import org.springframework.messaging.Message;
import org.springframework.messaging.PollableChannel;
import org.springframework.messaging.SubscribableChannel;
import lombok.extern.slf4j.Slf4j;
@SpringBootApplication
@IntegrationComponentScan
@Slf4j
public class App {
public static void main(String[] args) throws InterruptedException {
ConfigurableApplicationContext ctx = SpringApplication.run(App.class, args);
PollableChannel queueChannel = ctx.getBean("queueChannel", PollableChannel.class);
SubscribableChannel subscribableDirectChannel = ctx.getBean("subscribableDirectChannel", SubscribableChannel.class);
ScheduledExecutorService executor = Executors.newSingleThreadScheduledExecutor();
executor.scheduleWithFixedDelay(() -> {
log.warn("I am here!!!");
Message<?> receivedMessage = queueChannel.receive(5000);
if (isNull(receivedMessage)) {
return;
}
subscribableDirectChannel.send(receivedMessage);
}, 100, 300, TimeUnit.MILLISECONDS);
subscribableDirectChannel.subscribe((msg) -> log.warn("Receive msg: {}", msg));
log.warn("INIT");
for (int i = 0; i < 10; i++) {
queueChannel.send(MessageBuilder.withPayload("Start " + i).build());
}
log.warn("INIT FINISH");
Thread.sleep(5000);
log.warn("START");
log.warn("START FINISH");
log.warn("");
queueChannel.send(MessageBuilder.withPayload("Hello").build());
log.warn("");
queueChannel.send(MessageBuilder.withPayload("Hello2").build());
Thread.sleep(2_000);
log.warn("");
queueChannel.send(MessageBuilder.withPayload("Hello3").build());
Thread.sleep(3_000);
executor.shutdown();
}
@Bean
public PollableChannel queueChannel() {
return new QueueChannel(9);
}
@Bean
public DirectChannelSpec subscribableDirectChannel() {
return MessageChannels.direct("subscribableDirectChannel");
}
}
@@ -0,0 +1,5 @@
logging:
level:
root: info
org.springframework.integration: debug
@@ -0,0 +1,101 @@
package ru.otus.spring.integration;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import java.util.Collections;
import java.util.Map;
import org.junit.jupiter.api.Test;
import org.springframework.messaging.Message;
import org.springframework.messaging.MessageHeaders;
import org.springframework.messaging.support.ErrorMessage;
import org.springframework.messaging.support.GenericMessage;
import org.springframework.messaging.support.MessageBuilder;
@SuppressWarnings("all")
public class MessagesTest {
@Test
public void testCreateSimpleGenericMessage() {
// TODO: Создайте сообщение с payload-ом "Hello" с помощью конструктора
Message<String> message = new GenericMessage<>("Hello");
assertNotNull(message);
assertEquals(GenericMessage.class, message.getClass());
assertNotNull(message.getPayload());
assertEquals("Hello", message.getPayload());
}
@Test
public void testCreateGenericMessage() {
// TODO: Создайте сообщение с пользователем с помощью конструктора
Message<User> message = new GenericMessage<>(new User("John", 23));
assertNotNull(message);
assertEquals(GenericMessage.class, message.getClass());
assertNotNull(message.getPayload());
assertEquals(new User("John", 23), message.getPayload());
}
@Test
public void testGenericMessageWithHeaders() {
// TODO: Создайте сообщение с payload-ом "Hello" и header-ом "to":"World"
Map<String, Object> headers = Collections.singletonMap("to", "World");
Message<String> message = new GenericMessage<>("Hello", headers);
assertNotNull(message);
assertEquals("Hello", message.getPayload());
assertEquals("World", message.getHeaders().get("to", String.class));
}
@Test
public void testGenericMessageWithMessageHeaders() {
// TODO: Создайте сообщение с payload-ом "Hello" и header-ом "to":"World"
MessageHeaders headers = new MessageHeaders(Collections.singletonMap("to", "World"));
Message<String> message = new GenericMessage<>("Hello", headers);
assertNotNull(message);
assertEquals("Hello", message.getPayload());
assertEquals("World", message.getHeaders().get("to", String.class));
}
@Test
public void testErrorMessage() {
// TODO: Создайте сообщение об ошибки с объектом NullPointerException внутри
Message errorMessage = new ErrorMessage(new NullPointerException());
assertNotNull(errorMessage);
assertEquals(ErrorMessage.class, errorMessage.getClass());
assertEquals(NullPointerException.class, errorMessage.getPayload().getClass());
}
@Test
public void testMessageBuilder() {
// TODO: Создайте сообщение с payload-ом "Hello" и header-ом "to":"World" с помощью MessageBuilder
Message message = MessageBuilder.withPayload("Hello")
.setHeader("to", "World")
.build();
assertNotNull(message);
assertEquals("Hello", message.getPayload());
assertEquals("World", message.getHeaders().get("to", String.class));
}
@Test
public void testBuildFromMessage() {
Message<User> original = MessageBuilder
.withPayload(new User("Kate", 30))
.setHeader("processor", "userService")
.build();
// TODO: Создайте новое сообщение с теми же payload и header-ами c помощью MessageBuilder
Message<User> newMessage = MessageBuilder.fromMessage(original).build();
assertNotNull(newMessage);
assertEquals(original.getPayload(), newMessage.getPayload());
assertEquals(original.getHeaders().get("processor"), newMessage.getHeaders().get("processor"));
}
}
@@ -0,0 +1,35 @@
package ru.otus.spring.integration;
import java.util.Objects;
public class User {
private final String name;
private final int age;
public User(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public int getAge() {
return age;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (!(o instanceof User user)) return false;
return age == user.age &&
Objects.equals(name, user.name);
}
@Override
public int hashCode() {
return Objects.hash(name, age);
}
}
@@ -0,0 +1,29 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>3.2.5</version>
</parent>
<groupId>ru.otus</groupId>
<artifactId>integrations-channels</artifactId>
<version>1.0</version>
<packaging>pom</packaging>
<modules>
<module>integrations-channels-exercise</module>
<module>integrations-channels-solution</module>
</modules>
<properties>
<maven.compiler.target>17</maven.compiler.target>
<maven.compiler.source>17</maven.compiler.source>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
</project>