/** * Copyright (c) 2002-2012 "Neo Technology," * Network Engine for Objects in Lund AB [http://neotechnology.com] * * This file is part of Neo4j. * * Neo4j is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ package org.neo4j.consistency.checking; import java.util.Arrays; import org.neo4j.consistency.report.ConsistencyReport; import org.neo4j.consistency.store.DiffRecordAccess; import org.neo4j.consistency.store.RecordAccess; import org.neo4j.kernel.impl.nioneo.store.NeoStoreRecord; import org.neo4j.kernel.impl.nioneo.store.NodeRecord; import org.neo4j.kernel.impl.nioneo.store.PrimitiveRecord; import org.neo4j.kernel.impl.nioneo.store.PropertyRecord; import org.neo4j.kernel.impl.nioneo.store.Record; import org.neo4j.kernel.impl.nioneo.store.RelationshipRecord; public abstract class PrimitiveRecordCheck <RECORD extends PrimitiveRecord, REPORT extends ConsistencyReport.PrimitiveConsistencyReport<RECORD, REPORT>> implements RecordCheck<RECORD, REPORT> { private final RecordField<RECORD, REPORT>[] fields; public final ComparativeRecordChecker<RECORD, PrimitiveRecord, REPORT> ownerCheck = new ComparativeRecordChecker<RECORD, PrimitiveRecord, REPORT>() { @Override public void checkReference( RECORD record, PrimitiveRecord other, REPORT report, RecordAccess records ) { if ( other instanceof NodeRecord ) { report.multipleOwners( (NodeRecord) other ); } else if ( other instanceof RelationshipRecord ) { report.multipleOwners( (RelationshipRecord) other ); } else if ( other instanceof NeoStoreRecord ) { report.multipleOwners( (NeoStoreRecord) other ); } } }; PrimitiveRecordCheck( RecordField<RECORD, REPORT>... fields ) { this.fields = Arrays.copyOf( fields, fields.length + 1 ); this.fields[fields.length] = new FirstProperty(); } private class FirstProperty implements RecordField<RECORD, REPORT>, ComparativeRecordChecker<RECORD, PropertyRecord, REPORT> { @Override public void checkConsistency( RECORD record, REPORT report, RecordAccess records ) { if ( !Record.NO_NEXT_PROPERTY.is( record.getNextProp() ) ) { report.forReference( records.property( record.getNextProp() ), this ); } } @Override public long valueFrom( RECORD record ) { return record.getNextProp(); } @Override public void checkChange( RECORD oldRecord, RECORD newRecord, REPORT report, DiffRecordAccess records ) { if ( !newRecord.inUse() || valueFrom( oldRecord ) != valueFrom( newRecord ) ) { if ( !Record.NO_NEXT_PROPERTY.is( valueFrom( oldRecord ) ) && records.changedProperty( valueFrom( oldRecord ) ) == null ) { report.propertyNotUpdated(); } } } @Override public void checkReference( RECORD record, PropertyRecord property, REPORT report, RecordAccess records ) { if ( !property.inUse() ) { report.propertyNotInUse( property ); } else { if ( !Record.NO_PREVIOUS_PROPERTY.is( property.getPrevProp() ) ) { report.propertyNotFirstInChain( property ); } } } } @Override public void check( RECORD record, REPORT report, RecordAccess records ) { if ( !record.inUse() ) { return; } for ( RecordField<RECORD, REPORT> field : fields ) { field.checkConsistency( record, report, records ); } } @Override public void checkChange( RECORD oldRecord, RECORD newRecord, REPORT report, DiffRecordAccess records ) { check( newRecord, report, records ); if ( oldRecord.inUse() ) { for ( RecordField<RECORD, REPORT> field : fields ) { field.checkChange( oldRecord, newRecord, report, records ); } } } }