@ Конфигурируемый с модульным тестом вызывает ошибку
Классы составлены по времени компиляции. Я прочитал в документации, что сплетенные классы будут предупреждать о классах, созданных вне контекста контейнера, но не явной ошибке, но я получаю следующую тестовую ошибку в сборке maven:
testExecuteCommand(SportTimeExecutionCommandTest): Error creating bean with name 'SportTimeExecutionCommand': Injection of resource dependencies failed; nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: No bean named 'timeStrategyAnalyzer' is defined
И это несмотря на то, что я проводил тест с бегуном в классе дзюнитов.
@RunWith(MockitoJUnitRunner.class)
public class SportTimeExecutionCommandTest {
private static final Integer SPORT_ID = 1;
@Mock
private IdTimeRequestAdjuster idTimeRequestAdjuster ;
@Mock
private StrategyAnalyzer<List<TimeStep>> strategyAnalyzer;
@Mock
private SportStepExecutor stepExecutor ;
@Mock
private TimeRequestStrategy strategy ;
@Mock
private TimeStep step;
@Mock
private Future<List<EventData>> future;
private SportTimeExecutionCommand command;
private List<TimeStep> steps = new ArrayList<TimeStep>();
@Before
public void setUp() throws Exception {
command = new SportTimeExecutionCommand(SPORT_ID);
command.setIdTimeRequestAdjuster(idTimeRequestAdjuster);
command.setStepExecutor(stepExecutor);
command.setStrategyAnalyzer(strategyAnalyzer);
when(idTimeRequestAdjuster.getCurrentValue(SPORT_ID)).thenReturn(strategy);
when(strategyAnalyzer.calculateSteps(strategy)).thenReturn(steps);
steps.add(step);
when(stepExecutor.executeStep(SPORT_ID, step)).thenReturn(future);
when(future.get()).thenReturn(new ArrayList<EventData>());
}
@Test
public void testExecuteCommand() throws InterruptedException, ExecutionException {
command.executeCommand();
verify(idTimeRequestAdjuster).getCurrentValue(SPORT_ID);
verify(strategyAnalyzer).calculateSteps(strategy);
verify(stepExecutor).executeStep(SPORT_ID,step);
}
}
Класс реализации для справки:
@Configurable(autowire=Autowire.BY_TYPE)
public class SportTimeExecutionCommand implements AdjustingCommand<List<EventData>,Integer>{
private final static Logger logger = Logger.getLogger(SportTimeExecutionCommand.class.getName());
@Autowired
private IdTimeRequestAdjuster idTimeRequestAdjuster ;
@Resource(name = "timeStrategyAnalyzer")
private StrategyAnalyzer<List<TimeStep>> strategyAnalyzer;
@Autowired
private SportStepExecutor stepExecutor ;
private final Integer sportId ;
public SportTimeExecutionCommand(Integer sportId)
{
this.sportId = sportId ;
}
@Override
public List<EventData> executeCommand() throws InterruptedException, ExecutionException {
List<EventData> eventData = new ArrayList<EventData>();
List<TimeStep> timeSteps = strategyAnalyzer.calculateSteps(idTimeRequestAdjuster.getCurrentValue(sportId));
List<Future<List<EventData>>> futureResults = new ArrayList<Future<List<EventData>>>();
for (TimeStep timeStep : timeSteps) {
futureResults.add(stepExecutor.executeStep(sportId, timeStep));
}
for (Future<List<EventData>> futureEventData : futureResults) {
eventData.addAll(futureEventData.get());
}
return eventData;
}
@Override
public Integer getCriteria() {
return sportId;
}
@Override
public void adjust() {
logger.warning("adjusting sportId "+sportId+" value is now : "+idTimeRequestAdjuster.getCurrentValue(sportId).getRequests());
idTimeRequestAdjuster.adjustUp(sportId);
}
public void setIdTimeRequestAdjuster(IdTimeRequestAdjuster idTimeRequestAdjuster) {
this.idTimeRequestAdjuster = idTimeRequestAdjuster;
}
public void setStrategyAnalyzer(StrategyAnalyzer<List<TimeStep>> strategyAnalyzer) {
this.strategyAnalyzer = strategyAnalyzer;
}
public void setStepExecutor(SportStepExecutor stepExecutor) {
this.stepExecutor = stepExecutor;
}
}
Может кто-то видит проблему?
РЕДАКТИРОВАТЬ: я подозреваю, что он обрабатывает ресурсы в первую очередь, поэтому ошибка здесь происходит. Если я переключу аннотацию на автопроводку, то вместо этого она сломается на первом автопроводном компоненте.
2 ответа
Мое решение состояло в том, чтобы запустить отдельный профиль сборки maven, который компилируется без аспектов, а затем на этапе интеграции компилируется с аспектами.
Я думаю, что проблема вызвана name
атрибут в @Resource
аннотация:
@Resource(name = "timeStrategyAnalyzer")
private StrategyAnalyzer<List<TimeStep>> strategyAnalyzer;
Вероятно, если вы удалите его, внедрение зависимостей, выполняемое аспектами, будет работать нормально - обратите внимание, что на самом деле вы не раскрываете ресурс с этим именем нигде в опубликованном вами коде. Сплетенные классы, построенные вне Spring Container (используя new
оператор) также имеют право на внедрение зависимости.