[ ]:
# This information helps with debugging and getting support :)
import sys, platform
import pandas as pd
import bifacial_radiance as br
print("Working on a ", platform.system(), platform.release())
print("Python version ", sys.version)
print("Pandas version ", pd.__version__)
print("bifacial_radiance version ", br.__version__)

4 - Debugging with Custom Objects#

Fixed Tilt 2-up with Torque Tube + CLEAN Routine + CustomObject#

This journal has examples of various things, some which hav ebeen covered before and some in more depth:

  • Running a fixed_tilt simulation beginning to end.

  • Creating a 2-up module with torque-tube, and detailed geometry of spacings in xgap, ygap and zgap.

  • Calculating the tracker angle for a specific time, in case you want to use that value to model a fixed_tilt setup.

  • Loading and cleaning results, particularly important when using setups with torquetubes / ygaps.

  • Adding a “Custom Object” or marker at the Origin of the Scene, to do a visual sanity-check of the geometry.

It will look something like this (without the marker in this visualization):

What we are trying to re-create

STEPS:

  1. Specify Working Folder and Import Program

  2. Specify all variables

  3. Create the Radiance Object and generate the Sky

  4. Calculating tracker angle/geometry for a specific timestamp

  5. Making the Module & the Scene, Visualize and run Analysis

  6. Calculate Bifacial Ratio (clean results)

  7. Add Custom Elements to your Scene Example: Marker at 0,0 position

1. Specify Working Folder and Import Program#

[1]:
import os
from pathlib import Path

testfolder = Path().resolve().parent.parent / 'bifacial_radiance' / 'TEMP' / 'Tutorial_04'

# Another option using relative address; for some operative systems you might need '/' instead of '\'
# testfolder = os.path.abspath(r'..\..\bifacial_radiance\TEMP')

print ("Your simulation will be stored in %s" % testfolder)

if not os.path.exists(testfolder):
    os.makedirs(testfolder)

import bifacial_radiance
import numpy as np
import pandas as pd
Your simulation will be stored in C:\Users\mprillim\sam_dev\bifacial_radiance\bifacial_radiance\TEMP\Tutorial_04

2. Specify all variables for the module and scene#

Below find a list of all of the possible parameters for makeModule. scene and simulation parameters are also organized below. This simulation will be a complete simulation in terms of parameters that you can modify.

The below routine creates a HEXAGONAL torque tube, for a 2-UP configuration of a specific module size. Parameters for the module, the torque tube, and the scene are below. This is being run with gendaylit, for one specific timestamp

[2]:
simulationname = 'tutorial_4'

## SceneDict Parameters
gcr = 0.33   # ground cover ratio,  = module_height / pitch
albedo = 0.28  #'concrete'     # ground albedo
hub_height = 2.35  # we could also pass clearance_height.
azimuth_ang = 90 # Modules will be facing East.
lat = 37.5
lon = -77.6
nMods = 4   # doing a smaller array for better visualization on this example.
nRows = 2

# MakeModule Parameters
module_type='test-module'
x = 1.996      # landscape, sinze x > y. Remember that orientation has been deprecated.
y = 0.991
tilt = 10
numpanels = 2  # doing a 2-up system!


# Gaps:
xgap = 0.05  # distance between modules in the row.
ygap = 0.15  # distance between the 2 modules along the collector slope.
zgap = 0.175 # if there is a torquetube, this is the distance between the torquetube and the modules.
# If there is not a module, zgap is the distance between the module and the axis of rotation (relevant for
# tracking systems.

# TorqueTube Parameters
tubetype = 'Hex'
diameter = 0.15
material = 'Metal_Grey'       # IT's NOT GRAY, IT's GREY.

3. Create the Radiance Object and generate the Sky#

[3]:
demo = bifacial_radiance.RadianceObj(simulationname, path=str(testfolder))  # Create a RadianceObj 'object'
demo.setGround(albedo) # input albedo number or material name like 'concrete'.  To see options, run this without any input.
epwfile = demo.getEPW(lat,lon) # pull TMY data for any global lat/lon
metdata = demo.readWeatherFile(epwfile, coerce_year=2001) # read in the EPW weather data from above

timestamp = metdata.datetime.index(pd.to_datetime('2001-06-17 13:0:0 -5')) # Make this timezone aware, use -5 for EST.
demo.gendaylit(timestamp)  # Mid-day, June 17th
path = C:\Users\mprillim\sam_dev\bifacial_radiance\bifacial_radiance\TEMP\Tutorial_04
Making path: images
Making path: objects
Making path: results
Making path: skies
Making path: EPWs
Making path: materials
Loading albedo, 1 value(s), 0.280 avg
1 nonzero albedo values.
Getting weather file: USA_VA_Richmond.724010_TMY2.epw
 ... OK!
8760 line in WeatherFile. Assuming this is a standard hourly WeatherFile for the year for purposes of saving Gencumulativesky temporary weather files in EPW folder.
Coercing year to 2001
Saving file EPWs\metdata_temp.csv, # points: 8760
Calculating Sun position for Metdata that is right-labeled  with a delta of -30 mins. i.e. 12 is 11:30 sunpos
[3]:
'skies\\sky2_37.5_-77.33_2001-06-17_1300.rad'

4. Calculating tracker angle/geometry for a specific timestamp#

This trick is useful if you are trying to use the fixed-tilt steps in bifacial_radiance to model a tracker for one specific point in time (if you take a picture of a tracker, it looks fixed, right? Well then).

We assigned a 10 degree tilt at the beginning, but if we were to model a tracker as a fixed-tilt element because we are interested in only one point in time, this routine will tell us what tilt to use. Please note that to model a tracker as fixed tilt, we suggest passing a hub_height, otherwise you will have to calculate the clearance_height manually.

Details: you might have noticed in the previoust tutorial looking at the tracker dictionary, but the way that bifacial_radiance handles tracking: If the tracker is N-S axis azimuth, the surface azimuth of the modules will be set to 90 always, with a tilt that is either positive (for the early morning, facing East), or negative (for the afternoon, facing west).

[4]:
# Some tracking parameters that won't be needed after getting this angle:
axis_azimuth = 180
axis_tilt = 0
limit_angle = 60
backtrack = True
tilt = demo.getSingleTimestampTrackerAngle(metdata, timestamp, gcr, axis_azimuth, axis_tilt,limit_angle, backtrack)

print ("\n NEW Calculated Tilt: %s " % tilt)

 NEW Calculated Tilt: -4.67

5. Making the Module & the Scene, Visualize and run Analysis#

[5]:
# Making module with all the variables
module = demo.makeModule(name=module_type,x=x,y=y,bifi=1,
                         zgap=zgap, ygap=ygap,  xgap=xgap, numpanels=numpanels)
module.addTorquetube(diameter=diameter, material=material, tubetype=tubetype,
                    visible=True, axisofrotation=True)

# create a scene with all the variables.
# Specifying the pitch automatically with the collector width (sceney) returned by the module object.
# Height has been deprecated as an input. pass clearance_height or hub_height in the scenedict.

sceneDict = {'tilt':tilt,'pitch': np.round(module.sceney / gcr,3),
             'hub_height':hub_height,'azimuth':azimuth_ang,
             'module_type':module_type, 'nMods': nMods, 'nRows': nRows}

scene = demo.makeScene(module=module, sceneDict=sceneDict) #makeScene creates a .rad file of the Scene

octfile = demo.makeOct(demo.getfilelist())  # makeOct combines all of the ground, sky and object files into a .oct file.

Module Name: test-module
Module test-module updated in module.json
Module test-module updated in module.json
Pre-existing .rad file objects\test-module.rad will be overwritten

Created tutorial_4.oct

At this point you should be able to go into a command window (cmd.exe) and check the geometry. It should look like the image at the beginning of the journal. Example:

rvu -vf views:nbsphinx-math:`front`.vp -e .01 -pe 0.02 -vp -2 -12 14.5 tutorial_4.oct*

[6]:

## Comment the line below to run rvu from the Jupyter notebook instead of your terminal. ## Simulation will stop until you close the rvu window #!rvu -vf views\front.vp -e .01 tutorial_4.oct

And then proceed happily with your analysis:

[7]:
analysis = bifacial_radiance.AnalysisObj(octfile, demo.name)  # return an analysis object including the scan dimensions for back irradiance

sensorsy = 200 # setting this very high to see a detailed profile of the irradiance, including
#the shadow of the torque tube on the rear side of the module.
frontscan, backscan = analysis.moduleAnalysis(scene, modWanted = 2, rowWanted = 1, sensorsy = 200)
frontDict, backDict = analysis.analysis(octfile, demo.name, frontscan, backscan)  # compare the back vs front irradiance

# print('"Annual" bifacial ratio average:  %0.3f' %( sum(analysis.Wm2Back) / sum(analysis.Wm2Front) ) )
# See comment below of why this line is commented out.
Linescan in process: tutorial_4_Front
Linescan in process: tutorial_4_Back
Saved: results\irr_tutorial_4.csv

6. Calculate Bifacial Ratio (clean results)#

Although we could calculate a bifacial ratio average at this point, this value would be misleading, since some of the sensors generated will fall on the torque tube, the sky, and/or the ground since we have torquetube and ygap in the scene. To calculate the real bifacial ratio average, we must use the clean routines.

[8]:
resultFile='results/irr_tutorial_4.csv'
results_loaded = bifacial_radiance.load.read1Result(resultFile)
print("Printing the dataframe containing the results just calculated in %s: " % resultFile)
results_loaded
Printing the dataframe containing the results just calculated in results/irr_tutorial_4.csv:
[8]:
x y z rearZ mattype rearMat Wm2Front Wm2Back Back/FrontRatio
0 1.029825 6.305862e-17 2.706027 2.684100 a1.0.a0.test-module.6457 a1.0.a0.test-module.2310 911.9455 189.940467 0.208280
1 1.019254 6.241128e-17 2.705163 2.683236 a1.0.a0.test-module.6457 a1.0.a0.test-module.2310 911.9457 189.720333 0.208039
2 1.008682 6.176395e-17 2.704300 2.682373 a1.0.a0.test-module.6457 a1.0.a0.test-module.2310 911.9458 189.499700 0.207797
3 0.998110 6.111662e-17 2.703436 2.681509 a1.0.a0.test-module.6457 a1.0.a0.test-module.2310 911.9458 189.279033 0.207555
4 0.987538 6.046929e-17 2.702573 2.680646 a1.0.a0.test-module.6457 a1.0.a0.test-module.2310 911.9459 189.058400 0.207313
... ... ... ... ... ... ... ... ... ...
195 -1.031666 -6.317134e-17 2.537628 2.515701 a1.0.a1.test-module.6457 a1.0.a1.test-module.2310 911.9528 184.729067 0.202564
196 -1.042238 -6.381867e-17 2.536764 2.514838 a1.0.a1.test-module.6457 a1.0.a1.test-module.2310 911.9529 184.896467 0.202748
197 -1.052810 -6.446600e-17 2.535901 2.513974 a1.0.a1.test-module.6457 a1.0.a1.test-module.2310 911.9529 185.063900 0.202931
198 -1.063381 -6.511333e-17 2.535037 2.513110 a1.0.a1.test-module.6457 a1.0.a1.test-module.2310 911.9529 185.231333 0.203115
199 -1.073953 -6.576067e-17 2.534174 2.512247 a1.0.a1.test-module.6457 a1.0.a1.test-module.2310 911.9529 185.398733 0.203298

200 rows × 9 columns

[9]:
print("Looking at only 1 sensor in the middle -- position 100 out of the 200 sensors sampled:")
results_loaded.loc[100]
Looking at only 1 sensor in the middle -- position 100 out of the 200 sensors sampled:
[9]:
x                             -0.02735
y                                 -0.0
z                             2.619669
rearZ                         2.597742
mattype            a1.0.hextube1a.6457
rearMat                            sky
Wm2Front                    788.535333
Wm2Back                       163.0105
Back/FrontRatio               0.206725
Name: 100, dtype: object

As an example, we can see above that sensor 100 falls in the hextube, and in the sky. We need to remove this to calculate the real bifacial_gain from the irradiance falling into the modules. To do this we use cleanResult form the load.py module in bifacial_radiance. This finds the invalid materials and sets the irradiance values for those materials to NaN

This might take some time in the current version.

[10]:
# Cleaning Results:
# remove invalid materials and sets the irradiance values to NaN
clean_results = bifacial_radiance.load.cleanResult(results_loaded)
[11]:
print("Sampling the same location as before to see what the results are now:")
clean_results.loc[100]
Sampling the same location as before to see what the results are now:
[11]:
x                             -0.02735
y                                 -0.0
z                             2.619669
rearZ                         2.597742
mattype            a1.0.hextube1a.6457
rearMat                            sky
Wm2Front                           NaN
Wm2Back                            NaN
Back/FrontRatio               0.206725
Name: 100, dtype: object
[12]:
print('CORRECT Annual bifacial ratio average:  %0.3f' %( clean_results['Wm2Back'].sum() / clean_results['Wm2Front'].sum() ))

print ("\n(If we had not done the cleaning routine, the bifacial ratio would have been ", \
      "calculated to %0.3f  <-- THIS VALUE IS WRONG)" %( sum(analysis.Wm2Back) / sum(analysis.Wm2Front) ))

CORRECT Annual bifacial ratio average:  0.203

(If we had not done the cleaning routine, the bifacial ratio would have been  calculated to 0.205  <-- THIS VALUE IS WRONG)

7. Add Custom Elements to your Scene Example: Marker at 0,0 position#

This shows how to add a custom element, in this case a Cube, that will be placed in the center of your already created scene to mark the 0,0 location.

This can be added at any point after makeScene has been run once. Notice that if this extra element is in the scene and the analysis sensors fall on this element, they will measure irradiance at this element and no the modules.

We are going to create a “MyMarker.rad” file in the objects folder, right after we make the Module. This is a prism (so we use ‘genbox’), that is black from the ground.rad list of materials (‘black’) We are naming it ‘CenterMarker’ Its sides are going to be 0.5x0.5x0.5 m and We are going to leave its bottom surface coincident with the plane z=0, but going to center on X and Y.

[13]:
name='MyMarker'
text='! genbox black CenterMarker 0.1 0.1 4 | xform -t -0.05 -0.05 0'
customObject = demo.makeCustomObject(name,text)


Custom Object Name objects\MyMarker.rad

This should have created a MyMarker.rad object on your objects folder.

But creating the object does not automatically adds it to the seen. So let’s now add the customObject to the Scene. We are not going to translate it or anything because we want it at the center, but you can pass translation, rotation, and any other XFORM command from Radiance.

I am passing a rotation 0 because xform has to have something (I think) otherwise it gets confused.

[14]:
demo.appendtoScene(scene.radfiles, customObject, '!xform -rz 0')
# makeOct combines all of the ground, sky and object files into a .oct file.
octfile = demo.makeOct(demo.getfilelist())
Created tutorial_4.oct

appendtoScene appended to the Scene.rad file the name of the custom object we created and the xform transformation we included as text. Then octfile merged this new scene with the ground and sky files.

At this point you should be able to go into a command window (cmd.exe) and check the geometry, and the marker should be there.

[15]:

## Comment the line below to run rvu from the Jupyter notebook instead of your terminal. ## Simulation will stop until you close the rvu window #!rvu -vf views\front.vp -e .01 tutorial_4.oct

If you ran the getTrackerAngle detour and appended the marker, it should look like this:

Marker position at 0,0

If you do an analysis and any of the sensors hits the Box object we just created, the list of materials in the result.csv file should say something with “CenterMarker” on it.

See more examples of the use of makeCustomObject and appendtoScene on the Bifacial Carport/Canopies Tutorial