我正在尝试创建运行嵌入式Neo4J HA服务器的Java应用程序集群,但是在启动任何从属服务器时出现错误
Caused by: org.springframework.dao.InvalidDataAccessResourceUsageException: Error executing statement CREATE INDEX ON :`DeviceNode`(`deviceType`); nested exception is org.springframework.dao.InvalidDataAccessResourceUsageException: Error executing statement CREATE INDEX ON :`DeviceNode`(`deviceType`); nested exception is org.neo4j.cypher.CypherExecutionException: Modifying the database schema can only be done on the master server, this server is a slave. Please issue schema modification commands directly to the master.
at org.springframework.data.neo4j.support.query.CypherQueryEngine.query(CypherQueryEngine.java:56) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.support.schema.SchemaIndexProvider.createIndex(SchemaIndexProvider.java:36) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.support.mapping.EntityIndexCreator$1.doWithPersistentProperty(EntityIndexCreator.java:45) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.support.mapping.EntityIndexCreator$1.doWithPersistentProperty(EntityIndexCreator.java:41) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.mapping.model.BasicPersistentEntity.doWithProperties(BasicPersistentEntity.java:261) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.neo4j.support.mapping.EntityIndexCreator.ensureEntityIndexes(EntityIndexCreator.java:41) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.support.mapping.Neo4jMappingContext.updateStoredEntityType(Neo4jMappingContext.java:77) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.support.mapping.Neo4jMappingContext.addPersistentEntity(Neo4jMappingContext.java:71) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.support.mapping.Neo4jMappingContext.addPersistentEntity(Neo4jMappingContext.java:49) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.mapping.context.AbstractMappingContext.getPersistentEntity(AbstractMappingContext.java:170) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.mapping.context.AbstractMappingContext.getPersistentEntity(AbstractMappingContext.java:139) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.mapping.context.AbstractMappingContext.getPersistentEntity(AbstractMappingContext.java:65) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.neo4j.repository.query.CypherQueryBuilder.<init>(CypherQueryBuilder.java:37) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.repository.query.CypherQueryCreator.create(CypherQueryCreator.java:72) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.repository.query.CypherQueryCreator.create(CypherQueryCreator.java:35) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.repository.query.parser.AbstractQueryCreator.createCriteria(AbstractQueryCreator.java:109) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.repository.query.parser.AbstractQueryCreator.createQuery(AbstractQueryCreator.java:88) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.repository.query.parser.AbstractQueryCreator.createQuery(AbstractQueryCreator.java:73) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.neo4j.repository.query.DerivedCypherRepositoryQuery.<init>(DerivedCypherRepositoryQuery.java:59) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.repository.query.GraphQueryMethod.createQuery(GraphQueryMethod.java:146) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.neo4j.repository.GraphRepositoryFactory$1.resolveQuery(GraphRepositoryFactory.java:113) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.data.repository.core.support.RepositoryFactorySupport$QueryExecutorMethodInterceptor.<init>(RepositoryFactorySupport.java:304) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.repository.core.support.RepositoryFactorySupport.getRepository(RepositoryFactorySupport.java:161) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.repository.core.support.RepositoryFactoryBeanSupport.initAndReturn(RepositoryFactoryBeanSupport.java:224) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.repository.core.support.RepositoryFactoryBeanSupport.afterPropertiesSet(RepositoryFactoryBeanSupport.java:210) ~[spring-data-commons-1.7.0.RELEASE.jar:na]
at org.springframework.data.neo4j.repository.GraphRepositoryFactoryBean.afterPropertiesSet(GraphRepositoryFactoryBean.java:69) ~[spring-data-neo4j-3.0.0.RELEASE.jar:na]
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFactory.java:1571) ~[spring-beans-3.2.8.RELEASE.jar:3.2.8.RELEASE]
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1509) ~[spring-beans-3.2.8.RELEASE.jar:3.2.8.RELEASE]
... 38 common frames omitted
最重要的一点似乎是Modifying the database schema can only be done on the master server, this server is a slave. Please issue schema modification commands directly to the master
。我相信正在创建索引,因为SDN会解析Repository接口。
我在某个地方做过什么疯狂的事情,我可以告诉奴隶只是稍微退后一点,从主人那里拉出他们的架构,最终会到达那里吗?我相信我的配置相当标准:
<util:map id="config">
<entry key="org.neo4j.server.database.mode" value="HA"/>
<entry key="enable_remote_shell" value="true"/>
<entry key="ha.server_id" value="1"/>
<entry key="ha.initial_hosts" value="neo1:5001,neo2:5001,neo3:5001"/>
<entry key="ha.allow_init_cluster" value="true"/>
<entry key="ha.cluster_join_timeout" value="60s"/>
<entry key="ha.slave_only" value="false"/>
</util:map>
<bean id="graphDbFactory" class="org.neo4j.graphdb.factory.HighlyAvailableGraphDatabaseFactory"/>
<bean id="graphDbBuilder" factory-bean="graphDbFactory" factory-method="newHighlyAvailableDatabaseBuilder">
<constructor-arg value="#{neoloc}"/>
</bean>
<bean id="graphDbBuilderFinal" factory-bean="graphDbBuilder" factory-method="setConfig">
<constructor-arg ref="config"/>
</bean>
<bean id="graphDatabaseService" factory-bean="graphDbBuilderFinal" factory-method="newGraphDatabase" destroy-method="shutdown" />
<neo4j:config graphDatabaseService="graphDatabaseService"/>
<neo4j:repositories base-package="com.clarifimedia.data.repository"></neo4j:repositories>
<tx:annotation-driven transaction-manager="transactionManager"/>
<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
<property name="converters">
<set>
<bean class="com.clarifimedia.data.converter.DateTimeToLongConverter"/>
<bean class="com.clarifimedia.data.converter.LongToDateTimeConverter"/>
</set>
</property>
</bean>
配置值实际上是从JNDI读取的,但我上面有硬编码,因为#{xyz}
的负载不会那么方便。
版本
Neo4j的-2.0.1
Neo4j的-HA-2.0.1
弹簧数据的Neo4j-3.0.0.RELEASE
弹簧一般东西-3.2.8.RELEASE
[编辑,切线,错误?]
我试图通过破解网络来启动三个孤立的主人,目的是修复网络并查看选举是否发生。但是在启动时,由于TimeoutException,webapp无法启动。
Caused by: java.util.concurrent.TimeoutException: null
at org.neo4j.cluster.statemachine.StateMachineProxyFactory$ResponseFuture.get(StateMachineProxyFactory.java:300) ~[neo4j-cluster-2.0.1.jar:2.0.1]
at org.neo4j.cluster.client.ClusterJoin.joinByConfig(ClusterJoin.java:158) ~[neo4j-cluster-2.0.1.jar:2.0.1]
at org.neo4j.cluster.client.ClusterJoin.start(ClusterJoin.java:91) ~[neo4j-cluster-2.0.1.jar:2.0.1]
at org.neo4j.kernel.lifecycle.LifeSupport$LifecycleInstance.start(LifeSupport.java:503) ~[neo4j-kernel-2.0.1.jar:2.0.1]
ClusterJoin捕获InterruptedException和ExecutionException,但不捕获StateMachineProxyFactory $ ResponseFuture抛出的TimeoutException。鉴于ClusterJoin类中的while(true)
,我认为这是一个错误,或者我以某种方式混淆了jar文件(虽然这两个类都在neo4j-cluster-2.0.1中)。
答案 0 :(得分:3)
我有一个解决方法。请注意,我使用的是Spring Java配置,但通过XML进行配置应该不是什么大问题。
创建两个扩展标准sdn类的类,首先添加一个方法检查当前jvm是否以neo作为master运行:
public class OurDelegatingGraphDatabase extends DelegatingGraphDatabase {
public OurDelegatingGraphDatabase(GraphDatabaseService delegate) {
super(delegate);
}
public boolean isMaster(){
if (delegate instanceof HighlyAvailableGraphDatabase){
HighlyAvailableGraphDatabase highlyAvailableGraphDatabase = (HighlyAvailableGraphDatabase) delegate;
return highlyAvailableGraphDatabase.isMaster();
}
return true;
}
}
然后是一个自定义的SchemaIndexProvider,它只在主服务器上进行索引(单一模式也被认为是主服务器):
public class HACheckingSchemaIndexProvider extends SchemaIndexProvider {
private OurDelegatingGraphDatabase graphDatabase;
public HACheckingSchemaIndexProvider(GraphDatabase gd) {
super(gd);
if (gd instanceof OurDelegatingGraphDatabase){
this.graphDatabase = (OurDelegatingGraphDatabase) gd;
} else {
throw new IllegalStateException("OurDelegatingGraphDatabase was not configured");
}
}
@Override
public void createIndex(Neo4jPersistentProperty property) {
if (!graphDatabase.isMaster()){
return;
}
super.createIndex(property);
}
@Override
public void createIndex(String label, String prop, boolean unique) {
if (!graphDatabase.isMaster()){
return;
}
super.createIndex(label, prop, unique);
}
}
然后使用spring配置它们:
@Configuration
//..whatever else config you have
public class Neo4jConfig extends Neo4jConfiguration {
//other config stuf..
@Bean
@Autowired
@DependsOn("graphDatabaseService")
public GraphDatabase graphDatabase() {
return new OurDelegatingGraphDatabase(getGraphDatabaseService());
}
@Bean
@Override
public SchemaIndexProvider schemaIndexProvider() throws Exception {
return new HACheckingSchemaIndexProvider(graphDatabase());
}
答案 1 :(得分:2)
SDN不知道数据库是否为HA。
实际上我并不知道Slaves无法接受架构更新。
所以这可能是我们必须在Neo4j的spring配置上进行配置,以明确禁用某些设置的索引创建。
你能就此提出一个JIRA问题吗?
答案 2 :(得分:0)
对于那些急于将Wouter的Spring Java配置转换为XML等价的人,希望这会有所帮助(注意graphDatabaseService
被定义为OP):
<bean depends-on="graphDatabaseService" id="graphDatabase" class="com.path.to.OurDelegatingGraphDatabase">
<constructor-arg ref="graphDatabaseService" />
</bean>
<bean id="schemaIndexProvider" class="com.path.to.HACheckingSchemaIndexProvider">
<constructor-arg ref="graphDatabase" />
</bean>
这对我有用。当然,如果我犯了任何错误或上述内容可以改进,请告诉我。谢谢!