//This file is a part of G4MICE
//
//G4MICE is free software: you can redistribute it and/or modify
//it under the terms of the GNU General Public License as published by
//the Free Software Foundation, either version 3 of the License, or
//(at your option) any later version.
//
//G4MICE 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 General Public License for more details.
//
//You should have received a copy of the GNU General Public License
//along with G4MICE in the doc folder. If not, see
//.
#include "gtest/gtest.h"
#include "Geant4/G4Box.hh"
#include "Geant4/G4Tubs.hh"
#include "Geant4/G4Trd.hh"
#include "Geant4/G4Sphere.hh"
#include "Geant4/G4Polycone.hh"
#include "Geant4/G4Torus.hh"
#include "Geant4/G4EllipticalCone.hh"
#include "Geant4/G4UnionSolid.hh"
#include "Geant4/G4SubtractionSolid.hh"
#include "Geant4/G4IntersectionSolid.hh"
#include "Geant4/G4VSolid.hh"
#include "Geant4/G4AffineTransform.hh"
#include "EngModel/MiceModToG4Solid.hh"
#include "Utils/Exception.hh"
//MiceModToG4SolidTest
//Check that we can build each volume and try to check that we can't get any G4 exceptions
//i.e. that we catch all exceptions in G4MICE before G4 can get to them, and hopefully
//produce more polite error messages (that, for example, list the module name that saw the
//error.)
//
//TODO: boolean logical volume is not tested - needs external files; I think I would prefer to
// rework the way boolean logical volume works... need to think about it
//
namespace {
class MiceModToG4SolidTest : public ::testing::Test {
protected:
MiceModToG4SolidTest () {}
virtual ~MiceModToG4SolidTest() {}
virtual void SetUp () {}
virtual void TearDown() {}
};
TEST_F(MiceModToG4SolidTest, checkDimTest) {
MiceModule mod;
mod.setProperty("Dimensions", CLHEP::Hep3Vector(1.,1.,0.) );
ASSERT_THROW ( MiceModToG4Solid::checkDim(&mod, 3), MAUS::Exception );
ASSERT_NO_THROW( MiceModToG4Solid::checkDim(&mod, 2) );
}
TEST_F(MiceModToG4SolidTest, buildWedgeTest) {
MiceModule mod;
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1.,2.,3.) );
G4VSolid* sol=NULL;
ASSERT_NO_THROW( sol = MiceModToG4Solid::buildWedge(&mod) );
ASSERT_FALSE(sol==NULL);
EXPECT_DOUBLE_EQ(((G4Trd*)sol)->GetXHalfLength1(), 0.5); //half length of wedge width at bottom
EXPECT_DOUBLE_EQ(((G4Trd*)sol)->GetXHalfLength2(), 0.5); //half length of wedge width at top
EXPECT_DOUBLE_EQ(((G4Trd*)sol)->GetYHalfLength1(), 2.0);
EXPECT_NEAR (((G4Trd*)sol)->GetYHalfLength2(), 0.0, 1e-9);
EXPECT_DOUBLE_EQ(((G4Trd*)sol)->GetZHalfLength (), 1.5);
if (sol!=NULL) delete sol; sol=NULL;
mod.setProperty("Dimensions",CLHEP::Hep3Vector(0., 1., 1.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildWedge(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1., 0., 1.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildWedge(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1., 1., 0.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildWedge(&mod), MAUS::Exception );
}
MiceModule trapezoid_module( double aX1, double aX2, double aY1, double aY2, double aZ ) {
MiceModule mod;
try{
mod.setProperty("Dimensions",CLHEP::Hep3Vector(10.,20.,30.) );
mod.addPropertyDouble( "TrapezoidWidthX1", aX1 );
mod.addPropertyDouble( "TrapezoidWidthX2", aX2 );
mod.addPropertyDouble( "TrapezoidHeightY1", aY1 );
mod.addPropertyDouble( "TrapezoidHeightY2", aY2 );
mod.addPropertyDouble( "TrapezoidLengthZ", aZ );
return mod;
} catch( MAUS::Exception exc ){
std::string error = exc.GetMessage();
std::cerr << "Exception in trapezoid_module: " << error << std::endl;
throw( MAUS::Exception( MAUS::Exception::recoverable, "Error in MiceModToG4Test: " + error + "\'", "trapezoid_module"));
}
}
TEST_F(MiceModToG4SolidTest, buildTrapezoidTest) {
//
double x1 = 1.0, x2 = 2.0, y1 = 3.0, y2 = 4.0, z = 5.0;
MiceModule mod = trapezoid_module( x1, x2, y1, y2, z );
G4VSolid* sol=NULL;
ASSERT_NO_THROW( sol = MiceModToG4Solid::buildTrapezoid( &mod ) );
ASSERT_FALSE(sol==NULL);
EXPECT_DOUBLE_EQ(((G4Trd*)sol)->GetXHalfLength1(), x1 );
EXPECT_DOUBLE_EQ(((G4Trd*)sol)->GetXHalfLength2(), x2 );
EXPECT_DOUBLE_EQ(((G4Trd*)sol)->GetYHalfLength1(), y1 );
EXPECT_DOUBLE_EQ(((G4Trd*)sol)->GetYHalfLength2(), y2 );
EXPECT_DOUBLE_EQ(((G4Trd*)sol)->GetZHalfLength (), z );
if (sol!=NULL) delete sol; sol=NULL;
}
TEST_F(MiceModToG4SolidTest, buildBoxTest) {
MiceModule mod;
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1.,2.,3.) );
G4VSolid* sol=NULL;
ASSERT_NO_THROW( sol = MiceModToG4Solid::buildBox(&mod) );
ASSERT_FALSE(sol==NULL);
EXPECT_DOUBLE_EQ(((G4Box*)sol)->GetXHalfLength(), 0.5);
EXPECT_DOUBLE_EQ(((G4Box*)sol)->GetYHalfLength(), 1.0);
EXPECT_DOUBLE_EQ(((G4Box*)sol)->GetZHalfLength(), 1.5);
if (sol!=NULL) delete sol; sol=NULL;
mod.setProperty("Dimensions",CLHEP::Hep3Vector(0., 1., 1.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildBox(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1., 0., 1.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildBox(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1., 1., 0.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildBox(&mod), MAUS::Exception );
}
TEST_F(MiceModToG4SolidTest, buildCylinderTest) {
MiceModule mod;
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1.,2.,-1.) );
G4VSolid* sol=NULL;
ASSERT_NO_THROW( sol = MiceModToG4Solid::buildCylinder(&mod) );
ASSERT_FALSE(sol==NULL);
EXPECT_NEAR (((G4Tubs*)sol)->GetInnerRadius(), 0.0, 1e-9);
EXPECT_DOUBLE_EQ(((G4Tubs*)sol)->GetOuterRadius(), 1.0);
EXPECT_DOUBLE_EQ(((G4Tubs*)sol)->GetZHalfLength(), 1.0);
if (sol!=NULL) delete sol; sol=NULL;
mod.setProperty("Dimensions",CLHEP::Hep3Vector(0., 1., 1.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildCylinder(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1., 0., 1.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildCylinder(&mod), MAUS::Exception );
}
TEST_F(MiceModToG4SolidTest, buildTubeTest) {
MiceModule mod;
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1.,2.,3.) );
G4VSolid* sol=NULL;
ASSERT_NO_THROW( sol = MiceModToG4Solid::buildTube(&mod) );
ASSERT_FALSE(sol==NULL);
EXPECT_DOUBLE_EQ(((G4Tubs*)sol)->GetInnerRadius(), 1.0);
EXPECT_DOUBLE_EQ(((G4Tubs*)sol)->GetOuterRadius(), 2.0);
EXPECT_DOUBLE_EQ(((G4Tubs*)sol)->GetZHalfLength(), 1.5);
if (sol!=NULL) delete sol; sol=NULL;
//Check we catch errors before G4 throws
mod.setProperty("Dimensions",CLHEP::Hep3Vector(2., 1., 1.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildTube(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(0., 1., 1.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildTube(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1., 0., 1.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildTube(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1., 1., 0.) );
ASSERT_THROW ( sol = MiceModToG4Solid::buildTube(&mod), MAUS::Exception );
}
TEST_F(MiceModToG4SolidTest, buildSphereTest) {
MiceModule mod;
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1.,2.,-1.) );
G4VSolid* sol=NULL;
ASSERT_NO_THROW( sol = MiceModToG4Solid::buildSphere(&mod) );
ASSERT_FALSE(sol==NULL);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetInnerRadius(), 1.0);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetOuterRadius(), 2.0);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetStartPhiAngle(), 0.0);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetDeltaPhiAngle(), 360.0*deg);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetStartThetaAngle(), 0.0);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetDeltaThetaAngle(), 180.0*deg);
delete sol; sol=NULL;
mod.addPropertyHep3Vector("Phi", CLHEP::Hep3Vector(5.*deg, 10.*deg,-1.)); //radians < 2pi
mod.addPropertyHep3Vector("Theta", CLHEP::Hep3Vector(15.*deg,20.*deg,-1.)); //radians < pi
ASSERT_NO_THROW( sol = MiceModToG4Solid::buildSphere(&mod) );
ASSERT_FALSE(sol==NULL);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetInnerRadius(), 1.0);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetOuterRadius(), 2.0);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetStartPhiAngle(), 5.*deg);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetDeltaPhiAngle(), 10.*deg);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetStartThetaAngle(), 15.*deg);
EXPECT_DOUBLE_EQ(((G4Sphere*)sol)->GetDeltaThetaAngle(), 20.0*deg);
delete sol; sol=NULL;
//Check we catch errors before G4 throws
mod.setProperty("Dimensions",CLHEP::Hep3Vector(2.,1.,-1.) );
ASSERT_THROW( sol = MiceModToG4Solid::buildSphere(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1.,-1.,-1.) );
ASSERT_THROW( sol = MiceModToG4Solid::buildSphere(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(-1.,1.,-1.) );
ASSERT_THROW( sol = MiceModToG4Solid::buildSphere(&mod), MAUS::Exception );
mod.setProperty("Dimensions",CLHEP::Hep3Vector(1.,2.,-1.) );
//G4 throws if delta angle is negative, otherwise seems ok - but I want to force users to make valid input
mod.addPropertyHep3Vector("Phi", CLHEP::Hep3Vector(100.*deg, -10.*deg,-1.));
ASSERT_THROW( sol = MiceModToG4Solid::buildSphere(&mod), MAUS::Exception );
mod.addPropertyHep3Vector("Phi", CLHEP::Hep3Vector(500.*deg, 10.*deg,-1.));
ASSERT_THROW( sol = MiceModToG4Solid::buildSphere(&mod), MAUS::Exception );
mod.addPropertyHep3Vector("Phi", CLHEP::Hep3Vector(5.*deg, 500.*deg,-1.));
ASSERT_THROW( sol = MiceModToG4Solid::buildSphere(&mod), MAUS::Exception );
mod.addPropertyHep3Vector("Phi", CLHEP::Hep3Vector(5.*deg, 10.*deg,-1.));
mod.addPropertyHep3Vector("Theta", CLHEP::Hep3Vector(100.*deg, -10.*deg,-1.));
ASSERT_THROW( sol = MiceModToG4Solid::buildSphere(&mod), MAUS::Exception );
mod.addPropertyHep3Vector("Theta", CLHEP::Hep3Vector(500.*deg, 10.*deg,-1.));
ASSERT_THROW( sol = MiceModToG4Solid::buildSphere(&mod), MAUS::Exception );
mod.addPropertyHep3Vector("Theta", CLHEP::Hep3Vector(5.*deg, 500.*deg,-1.));
ASSERT_THROW( sol = MiceModToG4Solid::buildSphere(&mod), MAUS::Exception );
}
TEST_F(MiceModToG4SolidTest, buildPolyconeTest) {
MiceModule mod;
ASSERT_THROW(MiceModToG4Solid::buildPolycone(&mod), MAUS::Exception); //test properly in polycone.hh
}
TEST_F(MiceModToG4SolidTest, buildMultipoleTest) {
MiceModule mod;
ASSERT_THROW(MiceModToG4Solid::buildMultipole(&mod), MAUS::Exception); //test properly in multipole.hh
}
MiceModule torus_module(double in_r, double out_r, double q_s, double q_e, double r_curv) {
MiceModule mod;
mod.addPropertyDouble("TorusInnerRadius", in_r);
mod.addPropertyDouble("TorusOuterRadius", out_r);
mod.addPropertyDouble("TorusAngleStart", q_s);
mod.addPropertyDouble("TorusOpeningAngle", q_e);
mod.addPropertyDouble("TorusRadiusOfCurvature", r_curv);
return mod;
}
TEST_F(MiceModToG4SolidTest, buildTorusTest) {
MiceModule mod = torus_module(0.5, 1.0, 5.*deg, 10.*deg, 2.0);
G4Torus* sol=NULL;
ASSERT_NO_THROW(sol = (G4Torus*)MiceModToG4Solid::buildTorus(&mod));
ASSERT_FALSE(sol==NULL);
EXPECT_DOUBLE_EQ(sol->GetRmin(), 0.5 );
EXPECT_DOUBLE_EQ(sol->GetRmax(), 1.0 );
EXPECT_DOUBLE_EQ(sol->GetRtor(), 2.0 );
EXPECT_DOUBLE_EQ(sol->GetSPhi(), 5.*deg );
EXPECT_DOUBLE_EQ(sol->GetDPhi(), 10.*deg );
MiceModule mod1 = torus_module(1.5, 1.0, 5.*deg, 10.*deg, 2.0);
ASSERT_THROW(sol = (G4Torus*)MiceModToG4Solid::buildTorus(&mod1), MAUS::Exception);
MiceModule mod2 = torus_module(-0.1, 1.0, 5.*deg, 10.*deg, 2.0);
ASSERT_THROW(sol = (G4Torus*)MiceModToG4Solid::buildTorus(&mod2), MAUS::Exception);
MiceModule mod3 = torus_module(0.5, 3.0, 5.*deg, 10.*deg, 2.0);
ASSERT_THROW(sol = (G4Torus*)MiceModToG4Solid::buildTorus(&mod3), MAUS::Exception);
MiceModule mod4 = torus_module(0.5, 0.25, 5.*deg, 10.*deg, 2.0);
ASSERT_THROW(sol = (G4Torus*)MiceModToG4Solid::buildTorus(&mod4), MAUS::Exception);
MiceModule mod5 = torus_module(0.5, 1.0, -1.*deg, 10.*deg, 2.0);
ASSERT_THROW(sol = (G4Torus*)MiceModToG4Solid::buildTorus(&mod5), MAUS::Exception);
MiceModule mod6 = torus_module(0.5, 1.0, 361.*deg, 10.*deg, 2.0);
ASSERT_THROW(sol = (G4Torus*)MiceModToG4Solid::buildTorus(&mod6), MAUS::Exception);
MiceModule mod7 = torus_module(0.5, 1.0, 5.*deg, -1.*deg, 2.0);
ASSERT_THROW(sol = (G4Torus*)MiceModToG4Solid::buildTorus(&mod7), MAUS::Exception);
MiceModule mod8 = torus_module(0.5, 1.0, 5.*deg, 361.*deg, 2.0);
ASSERT_THROW(sol = (G4Torus*)MiceModToG4Solid::buildTorus(&mod8), MAUS::Exception);
}
TEST_F(MiceModToG4SolidTest, buildEllipticalConeTest) {
MiceModule mod;
mod.addPropertyDouble("ZCut", 1.0) ;
mod.setProperty("Dimensions",CLHEP::Hep3Vector(2.0, 3.0, 4.0));
G4EllipticalCone* sol=NULL;
ASSERT_NO_THROW(sol = (G4EllipticalCone*)MiceModToG4Solid::buildEllipticalCone(&mod));
ASSERT_FALSE(sol==NULL);
EXPECT_DOUBLE_EQ(sol->GetZTopCut(), 1.0);
EXPECT_DOUBLE_EQ(sol->GetSemiAxisMax(), 3.0/4.0);
MiceModule mod2;
mod2.addPropertyDouble("ZCut", -1.0) ;
mod2.setProperty("Dimensions",CLHEP::Hep3Vector(2.0, 3.0, 4.0));
ASSERT_THROW(sol = (G4EllipticalCone*)MiceModToG4Solid::buildEllipticalCone(&mod2), MAUS::Exception);
MiceModule mod3;
mod3.setProperty("Dimensions",CLHEP::Hep3Vector(-1.0, 3.0, 4.0));
ASSERT_THROW(sol = (G4EllipticalCone*)MiceModToG4Solid::buildEllipticalCone(&mod2), MAUS::Exception);
mod3.setProperty("Dimensions",CLHEP::Hep3Vector(1.0, -3.0, 4.0));
ASSERT_THROW(sol = (G4EllipticalCone*)MiceModToG4Solid::buildEllipticalCone(&mod2), MAUS::Exception);
mod3.setProperty("Dimensions",CLHEP::Hep3Vector(1.0, 3.0, -4.0));
ASSERT_THROW(sol = (G4EllipticalCone*)MiceModToG4Solid::buildEllipticalCone(&mod2), MAUS::Exception);
}
};