diff --git a/Project.toml b/Project.toml index 59b2e5d..5cb9be3 100644 --- a/Project.toml +++ b/Project.toml @@ -5,8 +5,6 @@ version = "0.1.0" [deps] CCBlade = "e1828068-15df-11e9-03e4-ef195ea46fa4" -CSV = "336ed68f-0bac-5ca0-87d4-7b16caf5d00b" -DataFrames = "a93c6f00-e57d-5684-b7b6-d8193f3e46c0" DelimitedFiles = "8bb1440f-4735-579b-a4ab-409b98df4dab" Distributed = "8ba89e20-285c-5b6f-9357-94700520ee1b" FLOWMath = "6cb5d3fb-0fe8-4cc2-bd89-9fe0b19a99d3" @@ -14,26 +12,20 @@ FiniteDiff = "6a86dc24-6348-571c-b903-95158fe2bd41" ForwardDiff = "f6369f11-7733-5829-9624-2563aa707210" Geodesy = "0ef565a4-170c-5f04-8de2-149903a85f3d" LinearAlgebra = "37e2e46d-f89d-539d-b4ee-838fcccc9c8e" -PyPlot = "d330b81b-6aea-500a-939a-2ce795aea3ee" SpecialFunctions = "276daf66-3868-5448-9aa4-cd146d93841b" TestSetExtensions = "98d24dd4-01ad-11ea-1b02-c9a08f80db04" -XLSX = "fdbf4ff8-1666-58a4-91e7-1b58723a45e0" YAML = "ddb6d928-2868-570f-bddf-ab3f9cf99eb6" [compat] -CSV = "0.10" +CCBlade = "0.2" +DelimitedFiles = "1" FLOWMath = "0.3" -TestSetExtensions = "3" -Geodesy = "1" -YAML = "0.4" -DataFrames = "1" -SpecialFunctions = "0,2" FiniteDiff = "2" -XLSX = "0.10" -DelimitedFiles = "1" -CCBlade = "0.2" -PyPlot = "2" ForwardDiff = "0.10" +Geodesy = "1" +SpecialFunctions = "0,2" +TestSetExtensions = "3" +YAML = "0.4" julia = "1" [extras] diff --git a/README.md b/README.md index 174bbbd..2dc6bd7 100644 --- a/README.md +++ b/README.md @@ -20,7 +20,8 @@ ### Install FLOWFarm -```julia +``` +julia (v1.x) pkg> dev https://github.com/byuflowlab/FLOWFarm.jl.git ``` @@ -28,7 +29,8 @@ To test FLOWFarm, run the following from the top directory: -```julia +``` +julia julia ] activate . diff --git a/docs/src/index.md b/docs/src/index.md index 212677a..ff9e8a1 100644 --- a/docs/src/index.md +++ b/docs/src/index.md @@ -9,13 +9,13 @@ - Smooth/continous model implementations - Runs on a single core, across multiple cores (threaded), or on multiple machines (distributed). - Designed so that new model implementations can be included by adding a single method -- Allows for Wake Expansion Continuation (WEC) as described [here](http://flowlab.groups.et.byu.net/preprints/Thomas2021.pdf) +- Allows for Wake Expansion Continuation (WEC) as described here ## Installation ### Install FLOWFarm - -```julia +``` +julia (v1.x) pkg> dev https://github.com/byuflowlab/FLOWFarm.jl.git ``` @@ -23,7 +23,8 @@ To test FLOWFarm, run the following from the top directory: -```julia +``` +julia julia ] activate . diff --git a/src/FLOWFarm.jl b/src/FLOWFarm.jl index 661522b..c675077 100644 --- a/src/FLOWFarm.jl +++ b/src/FLOWFarm.jl @@ -1,24 +1,14 @@ module FLOWFarm -using ForwardDiff: Iterators -using Geodesy: bound_thetad -using Geodesy; const gd = Geodesy using ForwardDiff using LinearAlgebra - -# using CCBlade -using PyPlot; const plt = PyPlot -using FLOWMath: linear,trapz,Akima,ksmax,abs_smooth -# using Statistics +using FLOWMath using Distributed -using YAML -using XLSX -using DataFrames -using CSV using SpecialFunctions +using Geodesy; const gd = Geodesy +using YAML include("io.jl") include("utilities.jl") -include("turbines.jl") include("windfarms.jl") include("wind_resources.jl") include("wind_shear_models.jl") @@ -31,8 +21,8 @@ include("general_models.jl") include("power_models.jl") include("user_functions.jl") include("optimization_functions.jl") -include("fatigue_model.jl") include("tip.jl") include("cost_models.jl") include("plotting.jl") +include("fatigue_model.jl") end # module diff --git a/src/cost_models.jl b/src/cost_models.jl index e7aa96c..6fd08dd 100644 --- a/src/cost_models.jl +++ b/src/cost_models.jl @@ -13,12 +13,12 @@ Container for parameters related to the Levelized Cost of Energy model (NREL 201 - `FCR::Float`: Fixed Charge Rate - `OpEx::Float`: Operational Expenditures """ -struct Levelized{TF} <: AbstractCostModel - TCC::TF - BOS::TF - FC::TF - FCR::TF - OpEx::TF +struct Levelized{T1,T2,T3,T4,T5} <: AbstractCostModel + TCC::T1 + BOS::T2 + FC::T3 + FCR::T4 + OpEx::T5 end Levelized() = Levelized(776.0, 326.0, 120.0, .0655, 43.0) # Default values taken from NREL 2019 Cost of Wind Energy @@ -37,7 +37,7 @@ Calculates the LCOE using the same numbers as NREL's FLORIS Model """ function cost_of_energy(rotor_diameter, hub_height, rated_power, AEP, Cost::Levelized) - + nturbines = length(rotor_diameter) # Taken from 2019 Cost of Wind Energy Review TCC = Cost.TCC diff --git a/src/fatigue_model.jl b/src/fatigue_model.jl index 0acba5b..68f0d27 100644 --- a/src/fatigue_model.jl +++ b/src/fatigue_model.jl @@ -1,8 +1,3 @@ -using FLOWFarm -using CCBlade - -const ff=FLOWFarm - """ multiple_components_op(U, V, W, Omega, r, precone, yaw, tilt, azimuth, rho, mu=1.81206e-05, asound=1.0) @@ -480,16 +475,16 @@ function get_single_damage(model_set,problem_description,turbine_ID,state_ID,nCy oms = zeros(nCycles*naz) turbine_inflow_velcities = zeros(nturbines) .+ ws - temp_resource = ff.DiscretizedWindResource([3*pi/2], [ws], [1.0], measurementheight, air_density,ambient_tis, wind_shear_model) - temp_pd = ff.WindFarmProblemDescription(windfarm, temp_resource, [windfarmstate]) - ff.turbine_velocities_one_direction!(points_x, points_y, model_set, temp_pd) + temp_resource = DiscretizedWindResource([3*pi/2], [ws], [1.0], measurementheight, air_density,ambient_tis, wind_shear_model) + temp_pd = WindFarmProblemDescription(windfarm, temp_resource, [windfarmstate]) + turbine_velocities_one_direction!(points_x, points_y, model_set, temp_pd) wfs_temp = temp_pd.wind_farm_states[1] for i = 1:nCycles*naz az = az_arr[(i+1)%naz+1] """need to figure this out""" - x_locs, y_locs, z_locs = ff.find_xyz_simple(turbine_x[turbine_ID],turbine_y[turbine_ID],turbine_z[turbine_ID].+hub_height,r,yaw,az) + x_locs, y_locs, z_locs = find_xyz_simple(turbine_x[turbine_ID],turbine_y[turbine_ID],turbine_z[turbine_ID].+hub_height,r,yaw,az) TI_arr = zeros(length(r)) for k = 1:length(r) @@ -518,7 +513,7 @@ function get_single_damage(model_set,problem_description,turbine_ID,state_ID,nCy U = U + turb_samples[i]*TI_inst.*U op = distributed_velocity_op.(U, Omega, r, precone, yaw, tilt, az, rho) out = CCBlade.solve.(Ref(rotor), sections, op) - flap[i],edge[i] = ff.get_moments(out,Rhub,Rtip,r,az,precone,tilt) + flap[i],edge[i] = get_moments(out,Rhub,Rtip,r,az,precone,tilt) oms[i] = Omega @@ -582,7 +577,7 @@ function get_single_state_damage(model_set,problem_description,state_ID,nCycles, nturbines = length(turbine_x) damage = zeros(nturbines) for k = 1:nturbines - damage[k] = ff.get_single_damage(model_set,problem_description,k,state_ID,nCycles,az_arr, + damage[k] = get_single_damage(model_set,problem_description,k,state_ID,nCycles,az_arr, turb_samples,points_x,points_y,omega_func,pitch_func,turbulence_func,r,rotor,sections,Rhub,Rtip,precone,tilt,rho, Nlocs=Nlocs,fos=fos) end @@ -607,11 +602,11 @@ function get_total_farm_damage(model_set,problem_description,nCycles,az_arr, for k = 1:ndirections problem_description.wind_farm_states[k].turbine_x[:],problem_description.wind_farm_states[k].turbine_y[:] = - ff.rotate_to_wind_direction(wind_farm.turbine_x, wind_farm.turbine_y, wind_resource.wind_directions[k]) + rotate_to_wind_direction(wind_farm.turbine_x, wind_farm.turbine_y, wind_resource.wind_directions[k]) problem_description.wind_farm_states[k].sorted_turbine_index[:] = sortperm(problem_description.wind_farm_states[k].turbine_x) - state_damage = ff.get_single_state_damage(model_set,problem_description,k,nCycles,az_arr, + state_damage = get_single_state_damage(model_set,problem_description,k,nCycles,az_arr, turb_samples,points_x,points_y,omega_func,pitch_func,turbulence_func,r,rotor,sections,Rhub,Rtip,precone,tilt,rho, Nlocs=Nlocs,fos=fos) damage = damage + state_damage.*frequencies[k] @@ -622,9 +617,9 @@ end -struct NpTp{T} - Np::T - Tp::T +struct NpTp{T1,T2} + Np::T1 + Tp::T2 end @@ -648,9 +643,9 @@ function get_single_damage_surr(turbine_x,turbine_y,turbine_z,rotor_diameter,hub edge = zeros(nCycles*naz) oms = zeros(nCycles*naz) - temp_resource = ff.DiscretizedWindResource([3*pi/2], [ws], [1.0], measurementheight, air_density, ambient_tis, wind_shear_model) + temp_resource = DiscretizedWindResource([3*pi/2], [ws], [1.0], measurementheight, air_density, ambient_tis, wind_shear_model) - turbine_velocities, turbine_ct, turbine_local_ti = ff.turbine_velocities_one_direction(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, + turbine_velocities, turbine_ct, turbine_local_ti = turbine_velocities_one_direction(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, turbine_ai, sorted_turbine_index, ct_model, rotor_sample_points_y, rotor_sample_points_z, wind_resource, model_set; wind_farm_state_id=state_ID) @@ -659,7 +654,7 @@ function get_single_damage_surr(turbine_x,turbine_y,turbine_z,rotor_diameter,hub az = az_arr[(i+1)%naz+1] """need to figure this out""" - x_locs, y_locs, z_locs = ff.find_xyz_simple(turbine_x[turbine_ID],turbine_y[turbine_ID],turbine_z[turbine_ID].+hub_height[turbine_ID],r,turbine_yaw[turbine_ID],az) + x_locs, y_locs, z_locs = find_xyz_simple(turbine_x[turbine_ID],turbine_y[turbine_ID],turbine_z[turbine_ID].+hub_height[turbine_ID],r,turbine_yaw[turbine_ID],az) TI_arr = zeros(length(r)) for k = 1:length(r) @@ -699,7 +694,7 @@ function get_single_damage_surr(turbine_x,turbine_y,turbine_z,rotor_diameter,hub out = NpTp(np,tp) # println("spline: ", out.Np[1], " ", out.Tp[1]) # println("___________________________") - flap[i],edge[i] = ff.get_moments(out,Rhub,Rtip,r,az,precone,tilt) + flap[i],edge[i] = get_moments(out,Rhub,Rtip,r,az,precone,tilt) oms[i] = Omega println("loop time: ", time()-time1) end diff --git a/src/general_models.jl b/src/general_models.jl index bb6082c..b22ffad 100644 --- a/src/general_models.jl +++ b/src/general_models.jl @@ -1,6 +1,4 @@ abstract type AbstractModelSet end -# using CSV -# using DataFrames """ WindFarmModelSet(wakedeficitmodel, wake_deflection_model, wake_combination_model, local_ti_model) @@ -32,20 +30,20 @@ Calculates the wind speed at a given point for a given state # Arguments - `loc::Array{TF,3}`: Location of interest -- `turbine_x::Array{TF,nTurbines}`: turbine east-west locations in the state +- `turbine_x::Array{TF,nTurbines}`: turbine east-west locations in the state reference frame -- `turbine_y::Array{TF,nTurbines}`: turbine north-south locations in the state +- `turbine_y::Array{TF,nTurbines}`: turbine north-south locations in the state reference frame - `turbine_z::Array{TF,nTurbines}`: turbine base height in the state reference frame -- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in +- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in radians - `turbine_ct::Array{TF,nTurbines}`: turbine thrust coefficients for the given state - `turbine_ai::Array{TF,nTurbines}`: turbine axial induction for the given state - `rotor_diameter::Array{TF,nTurbines}`: turbine rotor diameters - `hub_height::Array{TF,nTurbines}`: turbine hub heights -- `turbine_local_ti::Array{TF,nTurbines}`: turbine local turbulence intensity for +- `turbine_local_ti::Array{TF,nTurbines}`: turbine local turbulence intensity for the given state -- `sorted_turbine_index::Array{TF,nTurbines}`: array containing indices of wind turbines +- `sorted_turbine_index::Array{TF,nTurbines}`: array containing indices of wind turbines from most upwind to most downwind turbine in the given state - `wtvelocities::Array{TF,nTurbines}`: effective inflow wind speed for given state - `wind_resource::DiscretizedWindResource`: contains wind resource discreption (directions, @@ -59,12 +57,12 @@ function point_velocity(locx, locy, locz, turbine_x, turbine_y, turbine_z, turbi rotor_diameter, hub_height, turbine_local_ti, sorted_turbine_index, wtvelocities, wind_resource, model_set::AbstractModelSet; wind_farm_state_id=1, downwind_turbine_id=0) - + # extract flow information wind_speed = wind_resource.wind_speeds[wind_farm_state_id] reference_height = wind_resource.measurement_heights[wind_farm_state_id] - # set ground height + # set ground height ground_height = wind_resource.wind_shear_model.ground_height # TODO: allow topology to be given # find order for wind shear and deficit calculations @@ -127,7 +125,7 @@ function point_velocity(locx, locy, locz, turbine_x, turbine_y, turbine_z, turbi elseif shear_order == "first" point_velocity_out = point_velocity else - point_velocity_out = adjust_for_wind_shear(locz, point_velocity, reference_height, ground_height, wind_resource.wind_shear_model) + point_velocity_out = adjust_for_wind_shear(locz, point_velocity, reference_height, ground_height, wind_resource.wind_shear_model) end return point_velocity_out @@ -141,26 +139,26 @@ end turbine_ct=nothing, turbine_ai=nothing, turbine_local_ti=nothing) # Arguments -- `turbine_x::Array{TF,nTurbines}`: turbine east-west locations in the state +- `turbine_x::Array{TF,nTurbines}`: turbine east-west locations in the state reference frame -- `turbine_y::Array{TF,nTurbines}`: turbine north-south locations in the state +- `turbine_y::Array{TF,nTurbines}`: turbine north-south locations in the state reference frame - `turbine_z::Array{TF,nTurbines}`: turbine base height in the state reference frame - `rotor_diameter::Array{TF,nTurbines}`: turbine rotor diameters - `hub_height::Array{TF,nTurbines}`: turbine hub heights -- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in +- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in radians -- `sorted_turbine_index::Array{TF,nTurbines}`: turbine sorted order upstream to downstream +- `sorted_turbine_index::Array{TF,nTurbines}`: turbine sorted order upstream to downstream for given state -- `ct_model::AbstractThrustCoefficientModel`: defines how the thrust coefficient changes +- `ct_model::AbstractThrustCoefficientModel`: defines how the thrust coefficient changes with state etc -- rotor_sample_points_y::Array{TF,N}`: horizontal wind location of points to sample across - the rotor swept area when calculating the effective wind speed for the wind turbine. - Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) -- rotor_sample_points_z::Array{TF,N}`: vertical wind location of points to sample across the +- rotor_sample_points_y::Array{TF,N}`: horizontal wind location of points to sample across + the rotor swept area when calculating the effective wind speed for the wind turbine. + Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) +- rotor_sample_points_z::Array{TF,N}`: vertical wind location of points to sample across the rotor swept area when calculating the effective wind speed for the wind turbine. Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) -- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, +- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, frequencies, etc) - `model_set::AbstractModelSet`: defines wake-realated models to be used in analysis - `wind_farm_state_id::Int`: index to correct state to use from wind resource provided. @@ -170,7 +168,7 @@ function turbine_velocities_one_direction(turbine_x, turbine_y, turbine_z, rotor sorted_turbine_index, ct_model, rotor_sample_points_y, rotor_sample_points_z, wind_resource, model_set::AbstractModelSet; wind_farm_state_id::Int=1, velocity_only::Bool=true, turbine_velocities=nothing, turbine_ct=nothing, turbine_ai=nothing, turbine_local_ti=nothing) - + # get number of turbines and rotor sample point n_turbines = length(turbine_x) @@ -205,10 +203,10 @@ function turbine_velocities_one_direction(turbine_x, turbine_y, turbine_z, rotor turbine_ct, turbine_ai, turbine_local_ti; wind_farm_state_id=wind_farm_state_id, velocity_only=velocity_only) end - + if velocity_only - return turbine_velocities + return turbine_velocities else return turbine_velocities, turbine_ct, turbine_ai, turbine_local_ti end @@ -233,7 +231,7 @@ function turbine_velocities_one_direction!(turbine_x::Vector{T0}, turbine_y::Vec # initialize downstream wind turbine velocity to zero wind_turbine_velocity = 0.0 - # initialize point vel with shear + # initialize point vel with shear point_velocity_with_shear = 0.0 # loop over all rotor sample points to approximate the effective inflow velocity @@ -272,7 +270,7 @@ function turbine_velocities_one_direction!(turbine_x::Vector{T0}, turbine_y::Vec # get local turbulence intensity for this wind state ambient_ti = wind_resource.ambient_tis[wind_farm_state_id] - + # update local turbulence intensity for downstream turbine turbine_local_ti[downwind_turbine_id] = calculate_local_ti(turbine_x, turbine_y, ambient_ti, rotor_diameter, hub_height, turbine_yaw, turbine_local_ti, sorted_turbine_index, turbine_velocities, turbine_ct, model_set.local_ti_model; turbine_id=downwind_turbine_id, tol=1E-6) @@ -343,7 +341,7 @@ function turbine_velocities_one_direction_CC!(turbine_x::Vector{T0}, turbine_y:: # update local TI for current turbine turbine_local_ti[current_turbine_id] = calculate_local_ti(turbine_x, turbine_y, ambient_ti, rotor_diameter, hub_height, turbine_yaw, turbine_local_ti, sorted_turbine_index, - turbine_velocities, turbine_ct, model_set.local_ti_model; turbine_id=current_turbine_id, tol=1E-6) + turbine_velocities, turbine_ct, model_set.local_ti_model; turbine_id=current_turbine_id, tol=1E-6) for d = n+1:n_turbines downwind_turbine_id = Int(sorted_turbine_index[d]) @@ -376,7 +374,7 @@ function turbine_velocities_one_direction_CC!(turbine_x::Vector{T0}, turbine_y:: if p == 1 if no_yaw == false - deflections[current_turbine_id,downwind_turbine_id] = wake_deflection_model(x, y, z, turbine_x, turbine_yaw, turbine_ct, current_turbine_id, + deflections[current_turbine_id,downwind_turbine_id] = wake_deflection_model(x, y, z, turbine_x, turbine_yaw, turbine_ct, current_turbine_id, rotor_diameter, turbine_local_ti, model_set.wake_deflection_model) end sigma2[current_turbine_id,downwind_turbine_id] = wake_expansion(turbine_ct[current_turbine_id],turbine_local_ti[current_turbine_id],x_tilde_n,model_set.wake_deficit_model) @@ -444,7 +442,7 @@ end # n_turbines = Int(length(x)/2) # # println(typeof(x), n_turbines) -# turbine_x = x[1:n_turbines] +# turbine_x = x[1:n_turbines] # turbine_y = x[n_turbines+1:end] # # println(turbine_x) # # println("turbine_x type ", typeof(turbine_x)) @@ -513,7 +511,7 @@ end # end # if velocity_only -# return turbine_velocities +# return turbine_velocities # else # return turbine_velocities, turbine_ct, turbine_ai, turbine_local_ti # end @@ -524,8 +522,8 @@ end # model_set::AbstractModelSet, problem_description::AbstractWindFarmProblem; wind_farm_state_id=1) """ -calculate_flow_field(xrange, yrange, zrange, model_set::AbstractModelSet, turbine_x, - turbine_y, turbine_z, turbine_yaw, turbine_ct, turbine_ai, rotor_diameter, hub_height, +calculate_flow_field(xrange, yrange, zrange, model_set::AbstractModelSet, turbine_x, + turbine_y, turbine_z, turbine_yaw, turbine_ct, turbine_ai, rotor_diameter, hub_height, turbine_local_ti, sorted_turbine_index, wtvelocities, wind_resource; wind_farm_state_id=1) Generates a flow field for a given state and cross section @@ -535,23 +533,23 @@ Generates a flow field for a given state and cross section - `yrange::Range`: range defining north-west locations to sample in global reference frame - `zrange::Range`: range defining vertical locations to sample in global reference frame - `model_set::AbstractModelSet`: defines wake-realated models to be used in analysis -- `turbine_x::Array{TF,nTurbines}`: turbine east-west locations in the global +- `turbine_x::Array{TF,nTurbines}`: turbine east-west locations in the global reference frame -- `turbine_y::Array{TF,nTurbines}`: turbine north-south locations in the global +- `turbine_y::Array{TF,nTurbines}`: turbine north-south locations in the global reference frame - `turbine_z::Array{TF,nTurbines}`: turbine base height in the global reference frame -- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in +- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in radians - `turbine_ct::Array{TF,nTurbines}`: thrust coefficient of each turbine for the given state - `turbine_ai::Array{TF,nTurbines}`: turbine axial induction for the given state - `rotor_diameter::Array{TF,nTurbines}`: turbine rotor diameters - `hub_height::Array{TF,nTurbines}`: turbine hub heights -- `turbine_local_ti::Array{TF,nTurbines}`: turbine local turbulence intensity for +- `turbine_local_ti::Array{TF,nTurbines}`: turbine local turbulence intensity for the given state -- `sorted_turbine_index::Array{TF,nTurbines}`: turbine north-south locations in the +- `sorted_turbine_index::Array{TF,nTurbines}`: turbine north-south locations in the global reference frame - `wtvelocities::Array{TF,nTurbines}`: effective inflow wind speed for given state -- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, +- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, frequencies, etc) - `wind_farm_state_id::Int`: index to correct state to use from wind resource provided. Defaults to 1 @@ -591,7 +589,7 @@ function calculate_flow_field(xrange, yrange, zrange, end end - + # if zlen == 1 # return point_velocities[1,1:ylen,1:xlen] @@ -625,4 +623,4 @@ function calculate_flow_field(xrange, yrange, zrange, rotor_diameter, hub_height, turbine_local_ti, sorted_turbine_index, turbine_velocities, wind_resource, wind_farm_state_id=wind_farm_state_id) -end \ No newline at end of file +end diff --git a/src/io.jl b/src/io.jl index 5afdc42..c832735 100644 --- a/src/io.jl +++ b/src/io.jl @@ -20,7 +20,7 @@ function get_turb_loc_YAML(file_name; returnaep=false) nturbs = length(turb_coords) turbine_x = zeros(nturbs) turbine_y = zeros(nturbs) - + for i in 1:nturbs turbine_x[i] = turb_coords[i][1] turbine_y[i] = turb_coords[i][2] @@ -38,7 +38,7 @@ function get_turb_loc_YAML(file_name; returnaep=false) return turbine_x, turbine_y, fname_turb, fname_wr, AEP else return turbine_x, turbine_y, fname_turb, fname_wr - end + end end """ @@ -128,8 +128,8 @@ write turbine locations and related information to .yaml # Arguments - `file_name::String`: path/and/name/of/location/file.yaml """ -function write_turb_loc_YAML(filename, turbinex, turbiney; title="", titledescription="", - turbinefile="", locunits="m", wakemodelused="", windresourcefile="", aeptotal=[], +function write_turb_loc_YAML(filename, turbinex, turbiney; title="", titledescription="", + turbinefile="", locunits="m", wakemodelused="", windresourcefile="", aeptotal=[], aepdirs=[], aepunits="MWh", baseyaml=string(@__DIR__, "/default.yaml")) ### Retrieve turbine locations and auxiliary file names from <.yaml> file. @@ -146,23 +146,23 @@ function write_turb_loc_YAML(filename, turbinex, turbiney; title="", titledescri base["description"] = titledescription # save the title and description to the yaml database - base["definitions"]["plant_energy"]["properties"]["wake_model"]["items"][1]["\$ref"] = wakemodelused + base["definitions"]["plant_energy"]["properties"]["wake_model"]["items"][1]["\$ref"] = wakemodelused # save positions in yaml database turb_coords = fill(zeros(2), nturbines) for i in 1:nturbines turb_coords[i] = [turbinex[i], turbiney[i]] - end - + end + base["definitions"]["position"]["items"] = turb_coords - + # save the AEP in yaml database base["definitions"]["plant_energy"]["properties"]["annual_energy_production"]["default"] = aeptotal - # save the directional AEPs in the yaml database + # save the directional AEPs in the yaml database base["definitions"]["plant_energy"]["properties"]["annual_energy_production"]["binned"] = aepdirs - # save the directional AEP units in the yaml database + # save the directional AEP units in the yaml database base["definitions"]["plant_energy"]["properties"]["annual_energy_production"]["units"] = aepunits # save the auxiliary filenames for the windrose and the turbine attributes @@ -241,7 +241,7 @@ end get_boundary_yaml(filename) Returns the boundaries of a wind farm as defined in a yaml file in the format -used in FLOWFarm. Returns N by 2 array for single region farm and an array of +used in FLOWFarm. Returns N by 2 array for single region farm and an array of N by 2 arrays for multiple regions. Returned regions are sorted alphabetically by the keys provided in the yaml file. @@ -277,5 +277,5 @@ function get_boundary_yaml(filename) end return boundary_vertices - + end diff --git a/src/local_turbulence_intensity_models.jl b/src/local_turbulence_intensity_models.jl index 5f41c8f..c4a866f 100644 --- a/src/local_turbulence_intensity_models.jl +++ b/src/local_turbulence_intensity_models.jl @@ -14,7 +14,7 @@ end """ LocalTIModelMaxTI(astar, bstar, k1, k2) -Calculate local turbulence intensity using the model presented in Niayifar and +Calculate local turbulence intensity using the model presented in Niayifar and Porte Agel (2015, 2016) # Arguments @@ -23,11 +23,11 @@ Porte Agel (2015, 2016) - `k1::Float`: slope of k vs TI curve - `k2::Float`: vertical offset of k vs TI curve """ -struct LocalTIModelMaxTI{TF} <: AbstractLocalTurbulenceIntensityModel - astar::TF - bstar::TF - k1::TF - k2::TF +struct LocalTIModelMaxTI{T1,T2,T3,T4} <: AbstractLocalTurbulenceIntensityModel + astar::T1 + bstar::T2 + k1::T3 + k2::T4 end LocalTIModelMaxTI(x, y) = LocalTIModelMaxTI(x, y, 0.3837, 0.003678) LocalTIModelMaxTI() = LocalTIModelMaxTI(2.32, 0.154, 0.3837, 0.003678) @@ -35,7 +35,7 @@ LocalTIModelMaxTI() = LocalTIModelMaxTI(2.32, 0.154, 0.3837, 0.003678) """ LocalTIModelGaussTI() -Calculate local turbulence intensity using the model presented in Qian and +Calculate local turbulence intensity using the model presented in Qian and Ishihara (2018) """ @@ -50,16 +50,16 @@ end Returns ambient turbulence intesity value whenever local turbulence intensity is requestesd # Arguments -- `turbine_x::Array{Float,nTurbines}`: turbine wind direction locations in the wind direction +- `turbine_x::Array{Float,nTurbines}`: turbine wind direction locations in the wind direction reference frame -- `turbine_y::Array{Float,nTurbines}`: turbine cross wind locations in the wind direction +- `turbine_y::Array{Float,nTurbines}`: turbine cross wind locations in the wind direction reference frame - `ambient_ti::Float`: ambient turbulence intensity - `rotor_diameter::Array{Float,nTurbines}`: rotor diameters of all turbines - `hub_height::Array{Float,nTurbines}`: hub heights of all turbines relative to the ground - `turbine_yaw::Array{Float,nTurbines}`: yaw of all turbines for the current wind state in radians - `turbine_local_ti::Array{Float,nTurbines}`: local turbulence intensity of all turbines for the current wind state` -- `sorted_turbine_index::Array{Float,nTurbines}`: turbine north-south locations in the +- `sorted_turbine_index::Array{Float,nTurbines}`: turbine north-south locations in the global reference frame - `turbine_inflow_velcities::Array{Float,nTurbines}`: effective inflow wind speed at each turbine for given state - `turbine_ct::Array{Float,nTurbines}`: thrust coefficient of each turbine for the given state @@ -75,7 +75,7 @@ end """ _k_star_func(ti_ust,k1, k2) -Calculate local wake spreading rate based on turbulence intensity using the model presented +Calculate local wake spreading rate based on turbulence intensity using the model presented in Niayifar and Porte Agel (2015, 2016) # Arguments @@ -86,7 +86,7 @@ in Niayifar and Porte Agel (2015, 2016) # compute wake spread parameter based on local turbulence intensity function _k_star_func(ti_ust,k1,k2) # calculate wake spread parameter from Niayifar and Porte Agel (2015, 2016) - + k_star_ust = k1*ti_ust + k2 return k_star_ust @@ -95,7 +95,7 @@ end _k_star_func(x) = _k_star_func(x, 0.3837, 0.003678) """ - _niayifar_added_ti_function(x, d_dst, d_ust, h_ust, h_dst, ct_ust, kstar_ust, delta_y, + _niayifar_added_ti_function(x, d_dst, d_ust, h_ust, h_dst, ct_ust, kstar_ust, delta_y, ti_amb, ti_ust, ti_dst, ti_area_ratio_in; s=700.0) Main code for calculating the local turbulence intensity at a turbine using the method of @@ -141,7 +141,7 @@ function _niayifar_added_ti_function(x, d_dst, d_ust, h_ust, h_dst, ct_ust, ksta # only include turbines with area overlap in the softmax if wake_overlap > 0.0 # Calculate the turbulence added to the inflow of the downstream turbine by the - # wake of the upstream turbine based on Crespo, A.; Hernandez, J. Turbulence + # wake of the upstream turbine based on Crespo, A.; Hernandez, J. Turbulence # characteristics in wind-turbine wakes. J. Wind Eng. Ind. Aerodyn. 1996, # 61, 71–85. ti_added = 0.73*(axial_induction_ust^0.8325)*(ti_ust^0.0325)*((x/d_ust)^(-0.32)) @@ -172,16 +172,16 @@ end Returns local turbulence intensity calculated using Niayifar and Porte Agel 2015, 2016 using smooth max on area TI ratio # Arguments -- `turbine_x::Array{Float,nTurbines}`: turbine wind direction locations in the wind direction +- `turbine_x::Array{Float,nTurbines}`: turbine wind direction locations in the wind direction reference frame -- `turbine_y::Array{Float,nTurbines}`: turbine cross wind locations in the wind direction +- `turbine_y::Array{Float,nTurbines}`: turbine cross wind locations in the wind direction reference frame - `ambient_ti::Float`: ambient turbulence intensity - `rotor_diameter::Array{Float,nTurbines}`: rotor diameters of all turbines - `hub_height::Array{Float,nTurbines}`: hub heights of all turbines relative to the ground - `turbine_yaw::Array{Float,nTurbines}`: yaw of all turbines for the current wind state in radians - `turbine_local_ti::Array{Float,nTurbines}`: local turbulence intensity of all turbines for the current wind state` -- `sorted_turbine_index::Array{Float,nTurbines}`: turbine north-south locations in the +- `sorted_turbine_index::Array{Float,nTurbines}`: turbine north-south locations in the global reference frame - `turbine_inflow_velcities::Array{Float,nTurbines}`: effective inflow wind speed at each turbine for given state - `turbine_ct::Array{Float,nTurbines}`: thrust coefficient of each turbine for the given state @@ -234,12 +234,12 @@ function calculate_local_ti(turbine_x, turbine_y, ambient_ti, rotor_diameter, hu # calculate wake spread rate for current upstream turbine kstar_ust = _k_star_func(ti_ust,ti_model.k1,ti_model.k2) - # calculate the discontinuity point of the gauss yaw model + # calculate the discontinuity point of the gauss yaw model xd = _gauss_yaw_discontinuity(d_ust, x0, kstar_ust, kstar_ust, yaw_ust, ct_ust) - + # calculate horizontal wake spread sigmay = _gauss_yaw_spread_interpolated(d_ust, kstar_ust, x, x0, yaw_ust, xd) - + # calculate vertical wake spread sigmaz = _gauss_yaw_spread_interpolated(d_ust, kstar_ust, x, x0, 0.0, xd) @@ -271,24 +271,24 @@ function calculate_local_ti(turbine_x, turbine_y, ambient_ti, rotor_diameter, hu end """ - GaussianTI(loc,turbine_x, turbine_y, rotor_diameter, hub_height, turbine_ct, + GaussianTI(loc,turbine_x, turbine_y, rotor_diameter, hub_height, turbine_ct, sorted_turbine_index, ambient_ti; div_sigma=2.5, div_ti=1.2) -Calculate local turbulence intensity based on "On wake modeling, wind-farm gradients and AEP - predictions at the Anholt wind farm" by Pena Diaz, Alfredo; Hansen, Kurt Schaldemose; +Calculate local turbulence intensity based on "On wake modeling, wind-farm gradients and AEP + predictions at the Anholt wind farm" by Pena Diaz, Alfredo; Hansen, Kurt Schaldemose; Ott, Søren; van der Laan, Paul ?? # Arguments - `loc::Array{Float,3}`: [x,y,z] location of point of interest in wind direction ref. frame -- `turbine_x::Array{Float,nTurbines}`: turbine wind direction locations in the wind direction +- `turbine_x::Array{Float,nTurbines}`: turbine wind direction locations in the wind direction reference frame -- `turbine_y::Array{Float,nTurbines}`: turbine cross wind locations in the wind direction +- `turbine_y::Array{Float,nTurbines}`: turbine cross wind locations in the wind direction reference frame - `rotor_diameter::Array{Float,nTurbines}`: rotor diameters of all turbines - `hub_height::Array{Float,nTurbines}`: hub heights of all turbines relative to the ground - `turbine_ct::Array{Float,nTurbines}`: thrust coefficient of each turbine for the given state -- `sorted_turbine_index::Array{Float,nTurbines}`: turbine north-south locations in the +- `sorted_turbine_index::Array{Float,nTurbines}`: turbine north-south locations in the global reference frame - `ambient_ti::Float`: ambient turbulence intensity - `div_sigma::Float`: ? @@ -357,16 +357,16 @@ Returns local turbulence intensity calculated using methods in Qian 2018 from th with modification to account for yaw coming from Qian 2018 from Energies doi:10.3390/en11030665 # Arguments -- `turbine_x::Array{Float,nTurbines}`: turbine wind direction locations in the wind direction +- `turbine_x::Array{Float,nTurbines}`: turbine wind direction locations in the wind direction reference frame -- `turbine_y::Array{Float,nTurbines}`: turbine cross wind locations in the wind direction +- `turbine_y::Array{Float,nTurbines}`: turbine cross wind locations in the wind direction reference frame - `ambient_ti::Float`: ambient turbulence intensity - `rotor_diameter::Array{Float,nTurbines}`: rotor diameters of all turbines - `hub_height::Array{Float,nTurbines}`: hub heights of all turbines relative to the ground - `turbine_yaw::Array{Float,nTurbines}`: yaw of all turbines for the current wind state in radians - `turbine_local_ti::Array{Float,nTurbines}`: local turbulence intensity of all turbines for the current wind state` -- `sorted_turbine_index::Array{Float,nTurbines}`: turbine north-south locations in the +- `sorted_turbine_index::Array{Float,nTurbines}`: turbine north-south locations in the global reference frame - `turbine_inflow_velcities::Array{Float,nTurbines}`: effective inflow wind speed at each turbine for given state - `turbine_ct::Array{Float,nTurbines}`: thrust coefficient of each turbine for the given state @@ -422,7 +422,7 @@ function calculate_local_ti(turbine_x, turbine_y, ambient_ti, rotor_diameter, hu k2 = (cos(pi/2 * (r/D + 0.5)))^2 end sigma = k_star * dx + epsilon * D - + if dz >= 0 || hub_height[upstream_turbine] == 0 #to avoid divide by zero delta = 0 else diff --git a/src/plotting.jl b/src/plotting.jl index f8edeba..8a49f49 100644 --- a/src/plotting.jl +++ b/src/plotting.jl @@ -1,415 +1,404 @@ -# function circleshape(h, k, r) -# theta = LinRange(0, 2*pi, 500) -# return h .+ r*sin.(theta), k.+ r*cos.(theta) +# TODO: Replace this file using RecipesBase.jl + +# """ +# plotwindfarm!(ax, boundary_vertices, turbinex, turbiney, rotordiameter; nboundaries=1, aspect="equal", xlim=[], ylim=[], fill=false, color="k", markeralpha=1, title="") + +# Convenience function for plotting wind farms + +# # Arguments +# - `ax +# - `boundary_vertices::Array{Float,1}(nvertices)`: an nx2 array of boundary vertices +# - `turbinex::Array{Float,1}(nturbines)`: an array x coordinates of wind turbine locations +# - `turbiney::Array{Float,1}(nturbines)`: an array y coordinates of wind turbine locations +# - `rotordiameter::Array{Float,1}(nturbines)`: an array rotor diameters of wind turbines +# - `nboundaries::Int`: number of discrete boundary regions +# - `aspect::String`: set plot aspect ratio, default="equal" +# - `xlim::Array`: limits in x coordinate. "[]" results in limits being automatically defined +# - `ylim::Array`: limits in y coordinate. "[]" results in limits being automatically defined +# - `fill::Bool`: determines whether turbine circle markers are filled or not +# - `color::=String`: sets color for turbine markers +# - `markeralpha::Int`: determines tranparancy of turbine markers +# - `title::String`: optional title to include on the plot +# """ +# function plotwindfarm!(ax, turbinex, turbiney, rotordiameter; nboundaries=1, +# boundary_vertices=[], aspect="equal", xlim=[], ylim=[], fill=false, turbinecolor="k", +# boundarycolor="k", boundarylinestyle="-", turbinelinestyle="-", markeralpha=1, title="") + +# # determine how many turbines are in the farm +# nturbines = length(turbinex) + +# # add the boundary/ies +# if boundary_vertices != [] +# plotboundary!(ax, boundary_vertices, nboundaries=nboundaries, color=boundarycolor, linestyle=boundarylinestyle) +# end + +# # add the wind turbines +# plotlayout!(ax, turbinex, turbiney, rotordiameter; aspect=aspect, fill=fill, color=turbinecolor, markeralpha=markeralpha, title=title, linestyle=turbinelinestyle) + +# # adjust plot x limits if not set +# if xlim !== nothing +# if xlim == [] && boundary_vertices != [] +# if nboundaries == 1 +# xlim = [minimum(boundary_vertices[:,1]) - rotordiameter[1], maximum(boundary_vertices[:,1]) + rotordiameter[1]] +# else +# for i = 1:nboundaries +# xlim_tmp = [minimum(boundary_vertices[i][:,1]) - rotordiameter[1], maximum(boundary_vertices[i][:,1]) + rotordiameter[1]] +# if i == 1 +# xlim = deepcopy(xlim_tmp) +# end +# xlim[1] = minimum([xlim[1], xlim_tmp[1]]) +# xlim[2] = maximum([xlim[2], xlim_tmp[2]]) +# end +# end +# elseif boundary_vertices == [] +# xlim = [minimum(turbinex)-sum(rotordiameter)/nturbines, maximum(turbinex)+sum(rotordiameter)/nturbines] +# end +# ax.set(xlim=xlim) +# end + +# # adjust plot y limits if not set +# if ylim !== nothing +# if ylim == [] && boundary_vertices != [] +# if nboundaries == 1 +# ylim = [minimum(boundary_vertices[:,2]) - rotordiameter[1], maximum(boundary_vertices[:,2]) + rotordiameter[1]] +# else +# for i = 1:nboundaries +# ylim_tmp = [minimum(boundary_vertices[i][:,2]) - rotordiameter[1], maximum(boundary_vertices[i][:,2]) + rotordiameter[1]] +# if i == 1 +# ylim = deepcopy(ylim_tmp) +# end +# ylim[1] = minimum([ylim[1], ylim_tmp[1]]) +# ylim[2] = maximum([ylim[2], ylim_tmp[2]]) +# end +# end +# elseif boundary_vertices == [] +# ylim = [minimum(turbiney)-sum(rotordiameter)/nturbines, maximum(turbiney)+sum(rotordiameter)/nturbines] +# end +# ax.set(ylim=ylim) +# end + +# # adjust aspect ratio +# ax.set(aspect=aspect) # end -# function plotlayout!(p, turbinex, turbiney, rotordiameter; linecolor=:black, fillcolor=:blue, markeralpha=1, title="") +# """ +# plotlayout!(ax, turbinex, turbiney, rotordiameter; aspect="equal", xlim=[], ylim=[], fill=false, color="k", markeralpha=1, title="") + +# Convenience function for plotting wind farm layouts + +# # Arguments +# - `ax +# - `turbinex::Array{Float,1}(nturbines)`: an array x coordinates of wind turbine locations +# - `turbiney::Array{Float,1}(nturbines)`: an array y coordinates of wind turbine locations +# - `rotordiameter::Array{Float,1}(nturbines)`: an array rotor diameters of wind turbines +# - `aspect::String`: set plot aspect ratio, default="equal" +# - `xlim::Array`: limits in x coordinate. "[]" results in limits being automatically defined +# - `ylim::Array`: limits in y coordinate. "[]" results in limits being automatically defined +# - `fill::Bool`: determines whether turbine circle markers are filled or not +# - `color::=String`: sets color for turbine markers +# - `markeralpha::Int`: determines tranparancy of turbine markers +# - `itle::String`: optional title to include on the plot +# """ +# function plotlayout!(ax, turbinex, turbiney, rotordiameter; aspect="equal", fill=false, color="k", markeralpha=1, title="", linestyle="-") # nturbines = length(turbinex) + +# # add turbines # for i in 1:nturbines -# plot!(p, circleshape(turbinex[i],turbiney[i],rotordiameter[i]/2.0), seriestype=[:shape], -# linecolor=linecolor, c=fillcolor, legend=false, aspect_ratio=1, alpha=markeralpha, title=title) +# circle = matplotlib.patches.Circle((turbinex[i], turbiney[i]), rotordiameter[i]/2.0, fill=fill, color=color, linestyle=linestyle, zorder=Inf) +# ax.add_patch(circle) +# end + +# end + +# """ +# plotsingleboundary!(ax, boundary_vertices; color="k", linestyle="-') + +# Convenience function for plotting wind farm boundaries + +# # Arguments +# - `ax +# - `boundary_vertices::Array{Float,1}(nvertices)`: an nx2 array of boundary vertices for polygon or [[center_x, center_y], radius] for circle boundary +# - `color::=String`: sets color for turbine markers +# - `linestyle::String`: sets the line style for the boundary +# """ +# function plotsingleboundary!(ax, boundary_vertices; color="k", linestyle="-") + +# # get the number of vertices defining the boundary +# nvertices = length(boundary_vertices[:,1]) + +# # add boundary +# if nvertices < 3 +# # circle boundary +# farm_center = boundary_vertices[1] +# farm_radius = boundary_vertices[2] +# circle = matplotlib.patches.Circle((farm_center[1], farm_center[2]), farm_radius, color=color, linestyle=linestyle, fill=false) +# ax.add_patch(circle) +# else +# # polygon boundary +# x = push!(boundary_vertices[:,1], boundary_vertices[1,1]) +# y = push!(boundary_vertices[:,2], boundary_vertices[1,2]) +# ax.plot(x, y, color=color, linestyle=linestyle) +# end + +# end + +# """ +# plotboundary!(ax, boundary_vertices; color="k", linestyle="-", nboundaries=1) + +# Convenience function for plotting wind farm boundaries + +# # Arguments +# - `ax +# - `boundary_vertices::Array{Float,1}(nvertices)`: an nx2 array of boundary vertices for polygon or [[center_x, center_y], radius] for circle boundary +# - `color::=String`: sets color for turbine markers +# - `linestyle::String`: sets the line style for the boundary +# - `nboundaries::Int`: number of discrete boundary regions +# """ +# function plotboundary!(ax, boundary_vertices; color="k", linestyle="-", nboundaries=1) + +# if nboundaries == 1 +# plotsingleboundary!(ax, boundary_vertices, color=color, linestyle=linestyle) +# else +# for i = 1:nboundaries +# plotsingleboundary!(ax, boundary_vertices[i], color=color, linestyle=linestyle) +# end +# end + +# end + +# """ +# plotrotorsamplepoints!(ax, y, z; rotordiameter=2.0, aspect="equal", ylim=[], zlim=[], fill=false, color="k", markeralpha=1, title="") + +# Convenience function for plotting where points are being sampled on the wind turbine rotor + +# # Arguments +# - `ax +# - `y::Array{Float,1}(nturbines)`: an array x coordinates of wind turbine locations +# - `z::Array{Float,1}(nturbines)`: an array y coordinates of wind turbine locations +# - `rotordiameter::Number`: rotor diameter of wind turbine +# - `aspect::String`: set plot aspect ratio, default="equal" +# - `ylim::Array`: limits in y coordinate. "[]" results in limits being automatically defined +# - `zlim::Array`: limits in z coordinate. "[]" results in limits being automatically defined +# - `fill::Bool`: determines whether turbine circle markers are filled or not +# - `color::=String`: sets color for turbine markers +# - `markeralpha::Int`: determines tranparancy of turbine markers between 0 (transparent) and 1 (opaque) +# - `itle::String`: optional title to include on the plot +# """ +# function plotrotorsamplepoints!(ax, y, z; rotordiameter=2.0, aspect="equal", ylim=[], zlim=[], fill=false, color="k", markeralpha=1, title="") + +# # set plot limits +# if ylim == [] +# ylim = [-1.05, 1.05].*rotordiameter/2.0 +# end +# if zlim == [] +# zlim = [-1.05, 1.05].*rotordiameter/2.0 +# end + +# # add points +# ax.scatter(y, z, color=color, markeralpha) + +# # add rotor swept area +# circle = matplotlib.patches.Circle((0.0, 0.0), rotordiameter/2.0, fill=fill, color=color) +# ax.add_patch(circle) +# println(ylim, zlim) +# ax.set(xlim=ylim, ylim=zlim, aspect=aspect, title=title) + +# end + +# """ +# plotwindresource!(ax::Array, windresource::DiscretizedWindResource; roundingdigits=[1,3], fill=false, alpha=0.5, colors=["b", "b"], fontsize=8, edgecolor=nothing, rlabel_position=-45) + +# Convenience function for visualizing the wind speed and wind frequency roses + +# # Arguments +# - `ax::Array`: pre-initialized single dimension array of axes from pyplot with length at least 2 +# - `windresource::DiscretizedWindResource`: wind rose information +# - `roundingdigits::Array{Int, 1}`: how many significant digits to round to on each axis in ax +# - `fill::Bool`: determines whether bars are filled or not +# - `alpha::Number`: tranparancy of bars between 0 (transparent) and 1 (opaque) +# - `colors::=Array{String, 1}`: sets color for turbine markers +# - `fontsize::Int`: font size of text on figures +# - `edgecolor`: color of edges of each bar in polar chart, nothing means no color +# - `rlabel_position:Number`: Angle at which to draw the radial axes +# """ +# function plotwindresource!(windresource::DiscretizedWindResource; ax::Array=[], roundingdigits=[1,3], fill=false, alpha=0.5, colors=["b", "b"], fontsize=8, edgecolor=nothing, rlabel_position=-45, titles=["Wind Speed", "Wind Probability"]) + +# axes_generated = false +# if length(ax) == 0 +# fig, ax = plt.subplots(1, 2, subplot_kw=Dict("projection"=>"polar")) +# axes_generated = true +# end + +# # extract wind resource elements for windrose plots +# d = windresource.wind_directions +# s = windresource.wind_speeds +# f = windresource.wind_probabilities + +# # plot wind speed rose +# kwargs=(:edgecolor=>edgecolor, :alpha=>alpha, :color=>colors[1]) +# plotwindrose!(ax[1], d, s, roundingdigit=roundingdigits[1], fontsize=fontsize, units="m/s", title=titles[1], kwargs=kwargs) + +# # plot wind frequency rose +# kwargs=(:edgecolor=>edgecolor, :alpha=>alpha, :color=>colors[2]) +# plotwindrose!(ax[2], d, f, roundingdigit=roundingdigits[2], fontsize=fontsize, units="%", title=titles[2], kwargs=kwargs) + +# # return axis if newly generated +# axes_generated && return ax +# end + +# """ +# plotwindrose!(ax, d, f; roundingdigit=1, color="C0",alpha=0.5,fontsize=8, +# dticks=(0,pi/4,pi/2,3*pi/4,pi,5*pi/4,3*pi/2,7*pi/4), +# dlabels=("E","NE","N","NW","W","SW","S","SE"), +# fticks=nothing, flabels=nothing, normalize=false, edgecolor=nothing, units="", +# rlabel_position=-45) + +# Convenience function for creating a windrose from any polar data + +# # Arguments +# - `ax::PyCall.PyObject`: pre-initialized axis from pyplot +# - `d::Vector`: wind rose directions +# - `f::Vector`: wind rose radial variable +# - `roundingdigit::Int`: how many significant digits to round to +# - `fontsize::Int`: font size of text on figures +# - `dticks::Tuple`: contains angular tick locations +# - `dlabels::Tuple`: contains angular tick labels +# - `fticks::Tuple`: contains radial tick locations +# - `flabels::Tuple`: contains radial tick labels +# - `normalize::Bool`: choose whether or not to normalize by the sum of f +# - `units::String`: Units to append to flabels +# - `rlabel_position:Number`: Angle at which to draw the radial axis +# - `plotcommand::String`: Type of plot. Can be ["bar", "plot"] +# - `kwargs::Tuple`: tuple containing key word arguments to plotcommand in the form (key => "value") +# """ +# function plotwindrose!(ax, d, f; roundingdigit=1,fontsize=8, +# dticks=(0,pi/4,pi/2,3*pi/4,pi,5*pi/4,3*pi/2,7*pi/4), +# dlabels=("E","NE","N","NW","W","SW","S","SE"), dbuffer=0.3, +# fticks=nothing, flabels=nothing, normalize=false, units="", +# rlabel_position=-45, title="", plotcommand="bar", kwargs=(:edgecolor=>nothing, :alpha=>0.5, :color=>"C0")) + +# # set up function ticks if not provided, scale and round appropriately +# if fticks === nothing +# fmin = floor(minimum(f), digits=roundingdigit) +# fmax = ceil(maximum(f), digits=roundingdigit) +# frange = fmax - fmin +# if frange == 0.0 +# fticks = round.(collect((0:0.25:1.25).*fmax)[2:end-1], digits=roundingdigit) +# else +# fticks = round.(collect(fmin:frange/4.0:fmax)[2:end-1], digits=roundingdigit) +# end +# end +# # set up function labels if not provided +# if flabels === nothing +# flabels = string.(fticks).*" ".*units +# end + +# # normalize if desired +# if normalize +# f = f./sum(f) +# end + +# # adjust to radians if directions provided in degrees +# if maximum(d) > 10 +# d = deg2rad.(d) +# end + +# # get the number of wind directions +# ndirs = length(d) + +# # plot wind rose +# if plotcommand == "bar" +# # specify bar width +# width = (2*pi/ndirs)*0.95 +# ax.bar(pi/2 .-d, f; width, kwargs...) +# elseif plotcommand == "plot" +# ax.plot(pi/2 .-d, f; kwargs...) +# end + +# # format polar plot +# ax.set_xticks(dticks) +# ax.set_xticklabels(dlabels,fontsize=fontsize) +# ax.set_rgrids(fticks,flabels,angle=rlabel_position,fontsize=fontsize, zorder=0) +# # ax.set_thetagrids(dticks, dlabels, frac=1.0+dbuffer) +# for tick in ax.yaxis.get_majorticklabels() +# tick.set_horizontalalignment("center") +# end +# ax.set_title(title, y=-0.25,fontsize=fontsize) + +# end + +# """ +# add_turbine!(ax; view="side", hubdiameter=0.1, hubheight=0.9, radius=0.5, chord=0.1, +# nacellewidth=0.3, nacelleheight=0.1, towerbottomdiam=0.1, towertopdiam=0.05, +# overhang=0.05, s=5) + +# Convenience function for adding wind turbines to plots. + +# # Arguments +# - `ax::PyCall.PyObject`: pre-initialized axis from pyplot +# - `view::Number`: determines which turbine view to use "top" or "side" (default) +# - `hubdiameter::Number`: hub diameter in axis coordinate frame +# - `hubheight::Number`: hub height in axis coordinate frame +# - `radius::Number`: full rotor radius in axis coordinate frame +# - `chord::Number`: maximum chord in axis coordinate frame +# - `nacellewidth::Number`: nacelle width in axis coordinate frame +# - `nacelleheight::Number`: nacelle height in axis coordinate frame +# - `towerbottomdiam::Number`: tower bottom diameter in axis coordinate frame +# - `towertopdiam::Number`: tower top diameter in axis coordinate frame +# - `overhang::Number`: overhang (distance from blade attachment to tower bottom in x axis) in axis coordinate frame +# - `s::Number`: scales overhang and tower location in x direction to work with condensed x axis as in long contour plots +# """ +# function add_turbine!(ax; view="side", hubdiameter=0.1, hubheight=0.9, radius=0.5, chord=0.1, nacellewidth=0.3, nacelleheight=0.1, towerbottomdiam=0.1, towertopdiam=0.05, overhang=0.05, s=5, color="k") + +# if view == "side" + +# # create blade patches +# blade1 = plt.matplotlib.patches.Ellipse((0,hubheight+radius/2),chord, 0.5, color=color) +# blade2 = plt.matplotlib.patches.Ellipse((0,hubheight-radius/2),chord, 0.5, color=color) + +# # create hub and nacelle patches +# hub = plt.matplotlib.patches.Ellipse((0,hubheight),3*hubdiameter, hubdiameter, color=color) +# nacelle = plt.matplotlib.patches.Rectangle((0,hubheight-hubdiameter/2),nacellewidth, nacelleheight, color=color) + +# # scale if desired +# towerbottomdiam *= s +# towertopdiam *= s +# overhang *= s + +# # get difference between top and bottom tower diameters +# ddiff = abs(towertopdiam-towerbottomdiam) + +# # calculate polygon x points for tower +# p1x = overhang +# p2x = overhang+ddiff/2 +# p3x = p2x + towertopdiam +# p4x = p1x + towerbottomdiam + +# # create tower patch +# tower = plt.matplotlib.patches.Polygon([[p1x, 0.0],[p2x, hubheight],[p3x, hubheight],[p4x, 0.0]], closed=true, color=color) + +# # add patches to axis +# ax.add_patch(blade1) +# ax.add_patch(blade2) +# ax.add_patch(hub) +# ax.add_patch(nacelle) +# ax.add_patch(tower) + +# elseif view == "top" + +# # create blade patches +# blade1 = plt.matplotlib.patches.Ellipse((0,radius/2),chord, 0.5, color=color) +# blade2 = plt.matplotlib.patches.Ellipse((0,radius/2),chord, 0.5, color=color) + +# # create hub and nacelle patches +# hub = plt.matplotlib.patches.Ellipse((0.0,0.0),3*hubdiameter, hubdiameter, color=color) +# nacelle = plt.matplotlib.patches.Rectangle((0,-hubdiameter/2),nacellewidth, nacelleheight, color=color) + +# # add patches to axis +# ax.add_patch(blade1) +# ax.add_patch(blade2) +# ax.add_patch(hub) +# ax.add_patch(nacelle) + # end -# display(p) # end -""" - plotwindfarm!(ax, boundary_vertices, turbinex, turbiney, rotordiameter; nboundaries=1, aspect="equal", xlim=[], ylim=[], fill=false, color="k", markeralpha=1, title="") - - Convenience function for plotting wind farms - -# Arguments -- `ax -- `boundary_vertices::Array{Float,1}(nvertices)`: an nx2 array of boundary vertices -- `turbinex::Array{Float,1}(nturbines)`: an array x coordinates of wind turbine locations -- `turbiney::Array{Float,1}(nturbines)`: an array y coordinates of wind turbine locations -- `rotordiameter::Array{Float,1}(nturbines)`: an array rotor diameters of wind turbines -- `nboundaries::Int`: number of discrete boundary regions -- `aspect::String`: set plot aspect ratio, default="equal" -- `xlim::Array`: limits in x coordinate. "[]" results in limits being automatically defined -- `ylim::Array`: limits in y coordinate. "[]" results in limits being automatically defined -- `fill::Bool`: determines whether turbine circle markers are filled or not -- `color::=String`: sets color for turbine markers -- `markeralpha::Int`: determines tranparancy of turbine markers -- `title::String`: optional title to include on the plot -""" -function plotwindfarm!(ax, turbinex, turbiney, rotordiameter; nboundaries=1, - boundary_vertices=[], aspect="equal", xlim=[], ylim=[], fill=false, turbinecolor="k", - boundarycolor="k", boundarylinestyle="-", turbinelinestyle="-", markeralpha=1, title="") - - # determine how many turbines are in the farm - nturbines = length(turbinex) - - # add the boundary/ies - if boundary_vertices != [] - plotboundary!(ax, boundary_vertices, nboundaries=nboundaries, color=boundarycolor, linestyle=boundarylinestyle) - end - - # add the wind turbines - plotlayout!(ax, turbinex, turbiney, rotordiameter; aspect=aspect, fill=fill, color=turbinecolor, markeralpha=markeralpha, title=title, linestyle=turbinelinestyle) - - # adjust plot x limits if not set - if xlim !== nothing - if xlim == [] && boundary_vertices != [] - if nboundaries == 1 - xlim = [minimum(boundary_vertices[:,1]) - rotordiameter[1], maximum(boundary_vertices[:,1]) + rotordiameter[1]] - else - for i = 1:nboundaries - xlim_tmp = [minimum(boundary_vertices[i][:,1]) - rotordiameter[1], maximum(boundary_vertices[i][:,1]) + rotordiameter[1]] - if i == 1 - xlim = deepcopy(xlim_tmp) - end - xlim[1] = minimum([xlim[1], xlim_tmp[1]]) - xlim[2] = maximum([xlim[2], xlim_tmp[2]]) - end - end - elseif boundary_vertices == [] - xlim = [minimum(turbinex)-sum(rotordiameter)/nturbines, maximum(turbinex)+sum(rotordiameter)/nturbines] - end - ax.set(xlim=xlim) - end - - # adjust plot y limits if not set - if ylim !== nothing - if ylim == [] && boundary_vertices != [] - if nboundaries == 1 - ylim = [minimum(boundary_vertices[:,2]) - rotordiameter[1], maximum(boundary_vertices[:,2]) + rotordiameter[1]] - else - for i = 1:nboundaries - ylim_tmp = [minimum(boundary_vertices[i][:,2]) - rotordiameter[1], maximum(boundary_vertices[i][:,2]) + rotordiameter[1]] - if i == 1 - ylim = deepcopy(ylim_tmp) - end - ylim[1] = minimum([ylim[1], ylim_tmp[1]]) - ylim[2] = maximum([ylim[2], ylim_tmp[2]]) - end - end - elseif boundary_vertices == [] - ylim = [minimum(turbiney)-sum(rotordiameter)/nturbines, maximum(turbiney)+sum(rotordiameter)/nturbines] - end - ax.set(ylim=ylim) - end - - # adjust aspect ratio - ax.set(aspect=aspect) -end - -""" - plotlayout!(ax, turbinex, turbiney, rotordiameter; aspect="equal", xlim=[], ylim=[], fill=false, color="k", markeralpha=1, title="") - - Convenience function for plotting wind farm layouts - -# Arguments -- `ax -- `turbinex::Array{Float,1}(nturbines)`: an array x coordinates of wind turbine locations -- `turbiney::Array{Float,1}(nturbines)`: an array y coordinates of wind turbine locations -- `rotordiameter::Array{Float,1}(nturbines)`: an array rotor diameters of wind turbines -- `aspect::String`: set plot aspect ratio, default="equal" -- `xlim::Array`: limits in x coordinate. "[]" results in limits being automatically defined -- `ylim::Array`: limits in y coordinate. "[]" results in limits being automatically defined -- `fill::Bool`: determines whether turbine circle markers are filled or not -- `color::=String`: sets color for turbine markers -- `markeralpha::Int`: determines tranparancy of turbine markers -- `itle::String`: optional title to include on the plot -""" -function plotlayout!(ax, turbinex, turbiney, rotordiameter; aspect="equal", fill=false, color="k", markeralpha=1, title="", linestyle="-") - nturbines = length(turbinex) - - # add turbines - for i in 1:nturbines - circle = matplotlib.patches.Circle((turbinex[i], turbiney[i]), rotordiameter[i]/2.0, fill=fill, color=color, linestyle=linestyle, zorder=Inf) - ax.add_patch(circle) - end - -end - -""" - plotsingleboundary!(ax, boundary_vertices; color="k", linestyle="-') - - Convenience function for plotting wind farm boundaries - -# Arguments -- `ax -- `boundary_vertices::Array{Float,1}(nvertices)`: an nx2 array of boundary vertices for polygon or [[center_x, center_y], radius] for circle boundary -- `color::=String`: sets color for turbine markers -- `linestyle::String`: sets the line style for the boundary -""" -function plotsingleboundary!(ax, boundary_vertices; color="k", linestyle="-") - - # get the number of vertices defining the boundary - nvertices = length(boundary_vertices[:,1]) - - # add boundary - if nvertices < 3 - # circle boundary - farm_center = boundary_vertices[1] - farm_radius = boundary_vertices[2] - circle = matplotlib.patches.Circle((farm_center[1], farm_center[2]), farm_radius, color=color, linestyle=linestyle, fill=false) - ax.add_patch(circle) - else - # polygon boundary - x = push!(boundary_vertices[:,1], boundary_vertices[1,1]) - y = push!(boundary_vertices[:,2], boundary_vertices[1,2]) - ax.plot(x, y, color=color, linestyle=linestyle) - end - -end - -""" - plotboundary!(ax, boundary_vertices; color="k", linestyle="-", nboundaries=1) - - Convenience function for plotting wind farm boundaries - -# Arguments -- `ax -- `boundary_vertices::Array{Float,1}(nvertices)`: an nx2 array of boundary vertices for polygon or [[center_x, center_y], radius] for circle boundary -- `color::=String`: sets color for turbine markers -- `linestyle::String`: sets the line style for the boundary -- `nboundaries::Int`: number of discrete boundary regions -""" -function plotboundary!(ax, boundary_vertices; color="k", linestyle="-", nboundaries=1) - - if nboundaries == 1 - plotsingleboundary!(ax, boundary_vertices, color=color, linestyle=linestyle) - else - for i = 1:nboundaries - plotsingleboundary!(ax, boundary_vertices[i], color=color, linestyle=linestyle) - end - end - -end - -""" - plotrotorsamplepoints!(ax, y, z; rotordiameter=2.0, aspect="equal", ylim=[], zlim=[], fill=false, color="k", markeralpha=1, title="") - - Convenience function for plotting where points are being sampled on the wind turbine rotor - -# Arguments -- `ax -- `y::Array{Float,1}(nturbines)`: an array x coordinates of wind turbine locations -- `z::Array{Float,1}(nturbines)`: an array y coordinates of wind turbine locations -- `rotordiameter::Number`: rotor diameter of wind turbine -- `aspect::String`: set plot aspect ratio, default="equal" -- `ylim::Array`: limits in y coordinate. "[]" results in limits being automatically defined -- `zlim::Array`: limits in z coordinate. "[]" results in limits being automatically defined -- `fill::Bool`: determines whether turbine circle markers are filled or not -- `color::=String`: sets color for turbine markers -- `markeralpha::Int`: determines tranparancy of turbine markers between 0 (transparent) and 1 (opaque) -- `itle::String`: optional title to include on the plot -""" -function plotrotorsamplepoints!(ax, y, z; rotordiameter=2.0, aspect="equal", ylim=[], zlim=[], fill=false, color="k", markeralpha=1, title="") - - # set plot limits - if ylim == [] - ylim = [-1.05, 1.05].*rotordiameter/2.0 - end - if zlim == [] - zlim = [-1.05, 1.05].*rotordiameter/2.0 - end - - # add points - ax.scatter(y, z, color=color, markeralpha) - - # add rotor swept area - circle = matplotlib.patches.Circle((0.0, 0.0), rotordiameter/2.0, fill=fill, color=color) - ax.add_patch(circle) - println(ylim, zlim) - ax.set(xlim=ylim, ylim=zlim, aspect=aspect, title=title) - -end - -""" - plotwindresource!(ax::Array, windresource::ff.DiscretizedWindResource; roundingdigits=[1,3], fill=false, alpha=0.5, colors=["b", "b"], fontsize=8, edgecolor=nothing, rlabel_position=-45) - - Convenience function for visualizing the wind speed and wind frequency roses - -# Arguments -- `ax::Array`: pre-initialized single dimension array of axes from pyplot with length at least 2 -- `windresource::ff.DiscretizedWindResource`: wind rose information -- `roundingdigits::Array{Int, 1}`: how many significant digits to round to on each axis in ax -- `fill::Bool`: determines whether bars are filled or not -- `alpha::Number`: tranparancy of bars between 0 (transparent) and 1 (opaque) -- `colors::=Array{String, 1}`: sets color for turbine markers -- `fontsize::Int`: font size of text on figures -- `edgecolor`: color of edges of each bar in polar chart, nothing means no color -- `rlabel_position:Number`: Angle at which to draw the radial axes -""" -function plotwindresource!(windresource::ff.DiscretizedWindResource; ax::Array=[], roundingdigits=[1,3], fill=false, alpha=0.5, colors=["b", "b"], fontsize=8, edgecolor=nothing, rlabel_position=-45, titles=["Wind Speed", "Wind Probability"]) - - axes_generated = false - if length(ax) == 0 - fig, ax = plt.subplots(1, 2, subplot_kw=Dict("projection"=>"polar")) - axes_generated = true - end - - # extract wind resource elements for windrose plots - d = windresource.wind_directions - s = windresource.wind_speeds - f = windresource.wind_probabilities - - # plot wind speed rose - kwargs=(:edgecolor=>edgecolor, :alpha=>alpha, :color=>colors[1]) - plotwindrose!(ax[1], d, s, roundingdigit=roundingdigits[1], fontsize=fontsize, units="m/s", title=titles[1], kwargs=kwargs) - - # plot wind frequency rose - kwargs=(:edgecolor=>edgecolor, :alpha=>alpha, :color=>colors[2]) - plotwindrose!(ax[2], d, f, roundingdigit=roundingdigits[2], fontsize=fontsize, units="%", title=titles[2], kwargs=kwargs) - - # return axis if newly generated - axes_generated && return ax -end - -""" - plotwindrose!(ax, d, f; roundingdigit=1, color="C0",alpha=0.5,fontsize=8, - dticks=(0,pi/4,pi/2,3*pi/4,pi,5*pi/4,3*pi/2,7*pi/4), - dlabels=("E","NE","N","NW","W","SW","S","SE"), - fticks=nothing, flabels=nothing, normalize=false, edgecolor=nothing, units="", - rlabel_position=-45) - - Convenience function for creating a windrose from any polar data - -# Arguments -- `ax::PyCall.PyObject`: pre-initialized axis from pyplot -- `d::Vector`: wind rose directions -- `f::Vector`: wind rose radial variable -- `roundingdigit::Int`: how many significant digits to round to -- `fontsize::Int`: font size of text on figures -- `dticks::Tuple`: contains angular tick locations -- `dlabels::Tuple`: contains angular tick labels -- `fticks::Tuple`: contains radial tick locations -- `flabels::Tuple`: contains radial tick labels -- `normalize::Bool`: choose whether or not to normalize by the sum of f -- `units::String`: Units to append to flabels -- `rlabel_position:Number`: Angle at which to draw the radial axis -- `plotcommand::String`: Type of plot. Can be ["bar", "plot"] -- `kwargs::Tuple`: tuple containing key word arguments to plotcommand in the form (key => "value") -""" -function plotwindrose!(ax, d, f; roundingdigit=1,fontsize=8, - dticks=(0,pi/4,pi/2,3*pi/4,pi,5*pi/4,3*pi/2,7*pi/4), - dlabels=("E","NE","N","NW","W","SW","S","SE"), dbuffer=0.3, - fticks=nothing, flabels=nothing, normalize=false, units="", - rlabel_position=-45, title="", plotcommand="bar", kwargs=(:edgecolor=>nothing, :alpha=>0.5, :color=>"C0")) - - # set up function ticks if not provided, scale and round appropriately - if fticks === nothing - fmin = floor(minimum(f), digits=roundingdigit) - fmax = ceil(maximum(f), digits=roundingdigit) - frange = fmax - fmin - if frange == 0.0 - fticks = round.(collect((0:0.25:1.25).*fmax)[2:end-1], digits=roundingdigit) - else - fticks = round.(collect(fmin:frange/4.0:fmax)[2:end-1], digits=roundingdigit) - end - end - # set up function labels if not provided - if flabels === nothing - flabels = string.(fticks).*" ".*units - end - - # normalize if desired - if normalize - f = f./sum(f) - end - - # adjust to radians if directions provided in degrees - if maximum(d) > 10 - d = deg2rad.(d) - end - - # get the number of wind directions - ndirs = length(d) - - # plot wind rose - if plotcommand == "bar" - # specify bar width - width = (2*pi/ndirs)*0.95 - ax.bar(pi/2 .-d, f; width, kwargs...) - elseif plotcommand == "plot" - ax.plot(pi/2 .-d, f; kwargs...) - end - - # format polar plot - ax.set_xticks(dticks) - ax.set_xticklabels(dlabels,fontsize=fontsize) - ax.set_rgrids(fticks,flabels,angle=rlabel_position,fontsize=fontsize, zorder=0) - # ax.set_thetagrids(dticks, dlabels, frac=1.0+dbuffer) - for tick in ax.yaxis.get_majorticklabels() - tick.set_horizontalalignment("center") - end - ax.set_title(title, y=-0.25,fontsize=fontsize) - -end - -""" - add_turbine!(ax; view="side", hubdiameter=0.1, hubheight=0.9, radius=0.5, chord=0.1, - nacellewidth=0.3, nacelleheight=0.1, towerbottomdiam=0.1, towertopdiam=0.05, - overhang=0.05, s=5) - - Convenience function for adding wind turbines to plots. - -# Arguments -- `ax::PyCall.PyObject`: pre-initialized axis from pyplot -- `view::Number`: determines which turbine view to use "top" or "side" (default) -- `hubdiameter::Number`: hub diameter in axis coordinate frame -- `hubheight::Number`: hub height in axis coordinate frame -- `radius::Number`: full rotor radius in axis coordinate frame -- `chord::Number`: maximum chord in axis coordinate frame -- `nacellewidth::Number`: nacelle width in axis coordinate frame -- `nacelleheight::Number`: nacelle height in axis coordinate frame -- `towerbottomdiam::Number`: tower bottom diameter in axis coordinate frame -- `towertopdiam::Number`: tower top diameter in axis coordinate frame -- `overhang::Number`: overhang (distance from blade attachment to tower bottom in x axis) in axis coordinate frame -- `s::Number`: scales overhang and tower location in x direction to work with condensed x axis as in long contour plots -""" -function add_turbine!(ax; view="side", hubdiameter=0.1, hubheight=0.9, radius=0.5, chord=0.1, nacellewidth=0.3, nacelleheight=0.1, towerbottomdiam=0.1, towertopdiam=0.05, overhang=0.05, s=5, color="k") - - if view == "side" - - # create blade patches - blade1 = plt.matplotlib.patches.Ellipse((0,hubheight+radius/2),chord, 0.5, color=color) - blade2 = plt.matplotlib.patches.Ellipse((0,hubheight-radius/2),chord, 0.5, color=color) - - # create hub and nacelle patches - hub = plt.matplotlib.patches.Ellipse((0,hubheight),3*hubdiameter, hubdiameter, color=color) - nacelle = plt.matplotlib.patches.Rectangle((0,hubheight-hubdiameter/2),nacellewidth, nacelleheight, color=color) - - # scale if desired - towerbottomdiam *= s - towertopdiam *= s - overhang *= s - - # get difference between top and bottom tower diameters - ddiff = abs(towertopdiam-towerbottomdiam) - - # calculate polygon x points for tower - p1x = overhang - p2x = overhang+ddiff/2 - p3x = p2x + towertopdiam - p4x = p1x + towerbottomdiam - - # create tower patch - tower = plt.matplotlib.patches.Polygon([[p1x, 0.0],[p2x, hubheight],[p3x, hubheight],[p4x, 0.0]], closed=true, color=color) - - # add patches to axis - ax.add_patch(blade1) - ax.add_patch(blade2) - ax.add_patch(hub) - ax.add_patch(nacelle) - ax.add_patch(tower) - - elseif view == "top" - - # create blade patches - blade1 = plt.matplotlib.patches.Ellipse((0,radius/2),chord, 0.5, color=color) - blade2 = plt.matplotlib.patches.Ellipse((0,radius/2),chord, 0.5, color=color) - - # create hub and nacelle patches - hub = plt.matplotlib.patches.Ellipse((0.0,0.0),3*hubdiameter, hubdiameter, color=color) - nacelle = plt.matplotlib.patches.Rectangle((0,-hubdiameter/2),nacellewidth, nacelleheight, color=color) - - # add patches to axis - ax.add_patch(blade1) - ax.add_patch(blade2) - ax.add_patch(hub) - ax.add_patch(nacelle) - - end -end \ No newline at end of file diff --git a/src/power_models.jl b/src/power_models.jl index 7c1ef9b..6f5729c 100644 --- a/src/power_models.jl +++ b/src/power_models.jl @@ -25,9 +25,9 @@ Models will use adjust cp based on cp curve using linear interpolation of provid - `cp_points::Array{N,Float}`: power coefficient values corresponding to the provided speeds - 'pp::TF': exponent for adjusting power for wind turbine yaw """ -struct PowerModelCpPoints{ATF,TF} <: AbstractPowerModel - vel_points::ATF - cp_points::ATF +struct PowerModelCpPoints{ATF1,ATF2,TF} <: AbstractPowerModel + vel_points::ATF1 + cp_points::ATF2 pp::TF end PowerModelCpPoints(x,y) = PowerModelCpPoints(x, y, 2) @@ -35,7 +35,7 @@ PowerModelCpPoints(x,y) = PowerModelCpPoints(x, y, 2) """ PowerModelPowerPoints(vel_points, cp_points) -Models will use adjust wind turbine power based on power curve using linear interpolation of +Models will use adjust wind turbine power based on power curve using linear interpolation of provided points # Arguments @@ -43,9 +43,9 @@ provided points - `power_points::Array{N,Float}`: power values corresponding to the provided speeds - 'pp::TF': exponent for adjusting power for wind turbine yaw """ -struct PowerModelPowerPoints{ATF, TF} <: AbstractPowerModel - vel_points::ATF - power_points::ATF +struct PowerModelPowerPoints{ATF1,ATF2, TF} <: AbstractPowerModel + vel_points::ATF1 + power_points::ATF2 pp::TF end PowerModelPowerPoints(x,y) = PowerModelPowerPoints(x, y, 2) @@ -54,7 +54,7 @@ PowerModelPowerPoints(x,y) = PowerModelPowerPoints(x, y, 2) PowerModelPowerCurveCubic() Power will be calculated based on turbine specifications assuming a cubic power curve. Note -that this method is inherently incorrect and should only be used for theoretical purposes +that this method is inherently incorrect and should only be used for theoretical purposes or after careful validation. # Arguments @@ -204,7 +204,7 @@ Calculate the power for a wind turbine based on a pre-determined power curve wit defining the power curve """ function calculate_power(generator_efficiency, air_density, rotor_area, wt_velocity, wt_yaw, - cut_in_speed, rated_speed, cut_out_speed, rated_power, + cut_in_speed, rated_speed, cut_out_speed, rated_power, power_model::PowerModelPowerPoints; return_derivative=false) # get exponent for yaw adjustment @@ -263,7 +263,7 @@ Calculates wind turbine power using a cubic estimation based on turbine specific - `power_model::PowerModelPowerCurveCubic`: Empty struct """ function calculate_power(generator_efficiency, air_density, rotor_area, wt_velocity, wt_yaw, - cut_in_speed, rated_speed, cut_out_speed, rated_power, + cut_in_speed, rated_speed, cut_out_speed, rated_power, power_model::PowerModelPowerCurveCubic) pp = power_model.pp @@ -284,14 +284,14 @@ function calculate_power(generator_efficiency, air_density, rotor_area, wt_veloc end """ - calculate_turbine_power(generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, + calculate_turbine_power(generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, rotor_diameter, wt_velocity, power_model::AbstractPowerModel, air_density) Calculate the power for all wind turbines. Dispaches to desired power model. # Arguments - `generator_efficiency::Array{Float,nTurbines}` -- `cut_in_speed::Array{Float,nTurbines}` +- `cut_in_speed::Array{Float,nTurbines}` - `cut_out_speed::Array{Float,nTurbines}` - `rated_speed::Array{Float,nTurbines}` - `rated_power::Array{Float,nTurbines}` @@ -300,8 +300,8 @@ Calculate the power for all wind turbines. Dispaches to desired power model. - `power_model::AbstractPowerModel) - `air_density::Float` """ -function calculate_turbine_power(generator_efficiency, cut_in_speed, cut_out_speed, - rated_speed, rated_power, rotor_diameter, wt_velocity, wt_yaw, power_model::AbstractPowerModel, +function calculate_turbine_power(generator_efficiency, cut_in_speed, cut_out_speed, + rated_speed, rated_power, rotor_diameter, wt_velocity, wt_yaw, power_model::AbstractPowerModel, air_density; return_derivatives=false) # calculated wind turbine rotor-swept area @@ -317,14 +317,14 @@ function calculate_turbine_power(generator_efficiency, cut_in_speed, cut_out_spe end """ - turbine_powers_one_direction((generator_efficiency, cut_in_speed, cut_out_speed, + turbine_powers_one_direction((generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, rotor_diameter, turbine_inflow_velcities, air_density, power_model::AbstractPowerModel) Calculate the power for all wind turbines for a given state # Arguments - `generator_efficiency::Array{Float,nTurbines}` -- `cut_in_speed::Array{Float,nTurbines}` +- `cut_in_speed::Array{Float,nTurbines}` - `cut_out_speed::Array{Float,nTurbines}` - `rated_speed::Array{Float,nTurbines}` - `rated_power::Array{Float,nTurbines}` @@ -333,8 +333,8 @@ Calculate the power for all wind turbines for a given state - `air_density::Float` - `power_models::Array{nturbines})` elements of array should be be of sub-types or AbstractPowerModel """ -function turbine_powers_one_direction(generator_efficiency, cut_in_speed, cut_out_speed, - rated_speed, rated_power, rotor_diameter, turbine_inflow_velcities, turbine_yaw, air_density, +function turbine_powers_one_direction(generator_efficiency, cut_in_speed, cut_out_speed, + rated_speed, rated_power, rotor_diameter, turbine_inflow_velcities, turbine_yaw, air_density, power_models; jac=nothing) # get number of turbines and rotor sample point @@ -365,30 +365,30 @@ end Calculate power for each turbine for all states respectively # Arguments -- `turbine_x::Array{Float,nTurbines}`: turbine east-west locations in the global +- `turbine_x::Array{Float,nTurbines}`: turbine east-west locations in the global reference frame -- `turbine_y::Array{Float,nTurbines}`: turbine north-south locations in the global +- `turbine_y::Array{Float,nTurbines}`: turbine north-south locations in the global reference frame - `turbine_z::Array{Float,nTurbines}`: turbine base height in the global reference frame - `rotor_diameter::Array{Float,nTurbines}` - `hub_height::Array{TF,nTurbines}`: turbine hub heights -- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in +- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in radians -- `ct_model::AbstractThrustCoefficientModel`: defines how the thrust coefficient changes +- `ct_model::AbstractThrustCoefficientModel`: defines how the thrust coefficient changes with state etc - `generator_efficiency::Array{Float,nTurbines}` -- `cut_in_speed::Array{Float,nTurbines}` +- `cut_in_speed::Array{Float,nTurbines}` - `cut_out_speed::Array{Float,nTurbines}` - `rated_speed::Array{Float,nTurbines}` - `rated_power::Array{Float,nTurbines}` -- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, +- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, frequencies, etc) - `power_models::Array{nTurbines}`: elemenst of array should be sub-types of AbstractPowerModel - `model_set::AbstractModelSet`: defines wake-realated models to be used in analysis -- rotor_sample_points_y::Array{TF,N}`: horizontal wind location of points to sample across - the rotor swept area when calculating the effective wind speed for the wind turbine. - Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) -- rotor_sample_points_z::Array{TF,N}`: vertical wind location of points to sample across the +- rotor_sample_points_y::Array{TF,N}`: horizontal wind location of points to sample across + the rotor swept area when calculating the effective wind speed for the wind turbine. + Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) +- rotor_sample_points_z::Array{TF,N}`: vertical wind location of points to sample across the rotor swept area when calculating the effective wind speed for the wind turbine. Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) """ @@ -405,7 +405,7 @@ function calculate_state_turbine_powers(turbine_x, turbine_y, turbine_z, rotor_d arr_type = promote_type(typeof(turbine_x[1]),typeof(turbine_y[1]),typeof(turbine_z[1]),typeof(rotor_diameter[1]),typeof(hub_height[1]),typeof(turbine_yaw[1]), typeof(generator_efficiency[1]),typeof(cut_in_speed[1]),typeof(cut_out_speed[1]),typeof(rated_speed[1]),typeof(rated_power[1])) - + turbine_powers_by_direction = zeros(arr_type,(nstates, nturbines)) for i = 1:nstates @@ -422,7 +422,7 @@ function calculate_state_turbine_powers(turbine_x, turbine_y, turbine_z, rotor_d rated_power, rotor_diameter, turbine_velocities, turbine_yaw, wind_resource.air_density, power_models) turbine_powers_by_direction[i,:] = wt_power end - + return turbine_powers_by_direction end @@ -435,30 +435,30 @@ end Calculate AEP for each requested state respectively # Arguments -- `turbine_x::Array{Float,nTurbines}`: turbine east-west locations in the global +- `turbine_x::Array{Float,nTurbines}`: turbine east-west locations in the global reference frame -- `turbine_y::Array{Float,nTurbines}`: turbine north-south locations in the global +- `turbine_y::Array{Float,nTurbines}`: turbine north-south locations in the global reference frame - `turbine_z::Array{Float,nTurbines}`: turbine base height in the global reference frame - `rotor_diameter::Array{Float,nTurbines}` - `hub_height::Array{TF,nTurbines}`: turbine hub heights -- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in +- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in radians -- `ct_model::AbstractThrustCoefficientModel`: defines how the thrust coefficient changes +- `ct_model::AbstractThrustCoefficientModel`: defines how the thrust coefficient changes with state etc - `generator_efficiency::Array{Float,nTurbines}` -- `cut_in_speed::Array{Float,nTurbines}` +- `cut_in_speed::Array{Float,nTurbines}` - `cut_out_speed::Array{Float,nTurbines}` - `rated_speed::Array{Float,nTurbines}` - `rated_power::Array{Float,nTurbines}` -- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, +- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, frequencies, etc) - `power_model::Array{nTurbines}`: elemenst of array should be sub-types of AbstractPowerModel - `model_set::AbstractModelSet`: defines wake-realated models to be used in analysis -- rotor_sample_points_y::Array{TF,N}`: horizontal wind location of points to sample across - the rotor swept area when calculating the effective wind speed for the wind turbine. - Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) -- rotor_sample_points_z::Array{TF,N}`: vertical wind location of points to sample across the +- rotor_sample_points_y::Array{TF,N}`: horizontal wind location of points to sample across + the rotor swept area when calculating the effective wind speed for the wind turbine. + Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) +- rotor_sample_points_z::Array{TF,N}`: vertical wind location of points to sample across the rotor swept area when calculating the effective wind speed for the wind turbine. Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) - `hours_per_year::Float`: hours per year (averaged for leap year by default) @@ -474,34 +474,34 @@ function calculate_state_aeps(turbine_x, turbine_y, turbine_z, rotor_diameter, # set array type arr_type = promote_type(typeof(turbine_x[1]),typeof(turbine_y[1]),typeof(turbine_z[1]),typeof(rotor_diameter[1]),typeof(hub_height[1]),typeof(turbine_yaw[1]), typeof(generator_efficiency[1]),typeof(cut_in_speed[1]),typeof(cut_out_speed[1]),typeof(rated_speed[1]),typeof(rated_power[1])) - + # initialize state energy state_energy = zeros(arr_type,nstates) - # pre-allocate arrays for later calculations + # pre-allocate arrays for later calculations arr_type = promote_type(typeof(turbine_x[1]),typeof(turbine_y[1]),typeof(turbine_z[1]),typeof(rotor_diameter[1]), typeof(hub_height[1]),typeof(turbine_yaw[1])) n_turbines = length(turbine_x) - prealloc_turbine_velocities = zeros(arr_type, n_turbines) + prealloc_turbine_velocities = zeros(arr_type, n_turbines) prealloc_turbine_ct = zeros(arr_type, n_turbines) prealloc_turbine_ai = zeros(arr_type, n_turbines) prealloc_turbine_local_ti = zeros(arr_type, n_turbines) - + # loop over all states for i = 1:nstates - state_energy[i] = calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, + state_energy[i] = calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, ct_model, generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, power_models, rotor_sample_points_y, rotor_sample_points_z, wind_resource, model_set; wind_farm_state_id=i, hours_per_year=hours_per_year, weighted=weighted, prealloc_turbine_velocities=prealloc_turbine_velocities, prealloc_turbine_ct=prealloc_turbine_ct, prealloc_turbine_ai=prealloc_turbine_ai, prealloc_turbine_local_ti=prealloc_turbine_local_ti) - + end return state_energy end -function calculate_state_aep(turbine_x::Vector{T0}, turbine_y::Vector{T1}, turbine_z::Vector{T2}, rotor_diameter::Vector{T3}, hub_height::Vector{T4}, +function calculate_state_aep(turbine_x::Vector{T0}, turbine_y::Vector{T1}, turbine_z::Vector{T2}, rotor_diameter::Vector{T3}, hub_height::Vector{T4}, turbine_yaw::Vector{T5}, ct_model::Vector{<:AbstractThrustCoefficientModel}, generator_efficiency::Vector{T6}, cut_in_speed::Vector{T6}, cut_out_speed::Vector{T6}, rated_speed::Vector{T6}, rated_power::Vector{T6}, power_models::Vector{<:AbstractPowerModel}, rotor_sample_points_y::Vector{T6}, rotor_sample_points_z::Vector{T6}, wind_resource, model_set; wind_farm_state_id=1, hours_per_year=365.25*24.0, weighted=true, wind_speed_ids=nothing, prealloc_turbine_velocities=nothing, @@ -509,7 +509,7 @@ function calculate_state_aep(turbine_x::Vector{T0}, turbine_y::Vector{T1}, turbi # rotate turbine locations to match the direction of the current state rot_x, rot_y = rotate_to_wind_direction(turbine_x, turbine_y, wind_resource.wind_directions[wind_farm_state_id]) - + # get turbine indices in sorted order from upstream to downstream sorted_turbine_index = sortperm(rot_x) @@ -519,11 +519,11 @@ function calculate_state_aep(turbine_x::Vector{T0}, turbine_y::Vector{T1}, turbi sorted_turbine_index, ct_model, rotor_sample_points_y, rotor_sample_points_z, wind_resource, model_set; turbine_velocities=prealloc_turbine_velocities, turbine_ct=prealloc_turbine_ct, turbine_ai=prealloc_turbine_ai, turbine_local_ti=prealloc_turbine_local_ti, wind_farm_state_id=wind_farm_state_id, velocity_only=true) - + # calculate wind turbine powers for given state wt_power = turbine_powers_one_direction(generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, rotor_diameter, prealloc_turbine_velocities, turbine_yaw, wind_resource.air_density, power_models) - + # calculate wind farm power for given state state_power = sum(wt_power) @@ -555,14 +555,14 @@ function calculate_state_aep(turbine_x::Vector{T0}, turbine_y::Vector{T1}, turbi rated_power, rotor_diameter, prealloc_turbine_velocities, turbine_yaw, wind_resource.air_density, power_models) # sum the turbine powers to get the powers for the current speed/direction combination - state_power = sum(wt_power) + state_power = sum(wt_power) # add the weighted state power to the state (directional) aep state_aep += state_power*hours_per_year*wind_resource.wind_probabilities[wind_speed_ids[i]] end - # calculate weighted average for state power + # calculate weighted average for state power state_power = state_aep/hours_per_year end @@ -588,30 +588,30 @@ end Calculate wind farm AEP # Arguments -- `turbine_x::Array{Float,nTurbines}`: turbine east-west locations in the global +- `turbine_x::Array{Float,nTurbines}`: turbine east-west locations in the global reference frame -- `turbine_y::Array{Float,nTurbines}`: turbine north-south locations in the global +- `turbine_y::Array{Float,nTurbines}`: turbine north-south locations in the global reference frame - `turbine_z::Array{Float,nTurbines}`: turbine base height in the global reference frame - `rotor_diameter::Array{Float,nTurbines}` - `hub_height::Array{TF,nTurbines}`: turbine hub heights -- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in +- `turbine_yaw::Array{TF,nTurbines}`: turbine yaw for the given wind direction in radians -- `ct_model::AbstractThrustCoefficientModel`: defines how the thrust coefficient changes +- `ct_model::AbstractThrustCoefficientModel`: defines how the thrust coefficient changes with state etc - `generator_efficiency::Array{Float,nTurbines}` -- `cut_in_speed::Array{Float,nTurbines}` +- `cut_in_speed::Array{Float,nTurbines}` - `cut_out_speed::Array{Float,nTurbines}` - `rated_speed::Array{Float,nTurbines}` - `rated_power::Array{Float,nTurbines}` -- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, +- `wind_resource::DiscretizedWindResource`: wind resource discreption (directions, speeds, frequencies, etc) - `power_model::Array{)`: elements of array should be sub types of AbstractPowerModel - `model_set::AbstractModelSet`: defines wake-realated models to be used in analysis -- `rotor_sample_points_y::Array{TF,N}`: horizontal wind location of points to sample across - the rotor swept area when calculating the effective wind speed for the wind turbine. - Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) -- `rotor_sample_points_z::Array{TF,N}`: vertical wind location of points to sample across the +- `rotor_sample_points_y::Array{TF,N}`: horizontal wind location of points to sample across + the rotor swept area when calculating the effective wind speed for the wind turbine. + Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) +- `rotor_sample_points_z::Array{TF,N}`: vertical wind location of points to sample across the rotor swept area when calculating the effective wind speed for the wind turbine. Points are centered at the hub (0,0) and scaled by the radius (1=tip of blades) - `hours_per_year::Float`: hours per year (averaged for leap year by default) @@ -624,15 +624,15 @@ function calculate_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, # find how many wind states are being calculated nstates = length(wind_resource.wind_directions) - # check if we can run a single wind speed for each direction to save time + # check if we can run a single wind speed for each direction to save time if typeof(model_set.wake_combination_model) == SumOfSquaresFreestreamSuperposition # find unique directions unique_directions = unique(wind_resource.wind_directions) - # find how many unique directions there are + # find how many unique directions there are ndirections = length(unique_directions) end - + # state_energy = Vector{typeof(wind_farm.turbine_x[1])}(undef,nstates) arr_type = promote_type(typeof(turbine_x[1]),typeof(turbine_y[1]),typeof(turbine_z[1]),typeof(rotor_diameter[1]),typeof(hub_height[1]),typeof(turbine_yaw[1]), typeof(generator_efficiency[1]),typeof(cut_in_speed[1]),typeof(cut_out_speed[1]),typeof(rated_speed[1]),typeof(rated_power[1])) @@ -648,9 +648,9 @@ function calculate_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, # take a speed in the middle (so it is not zero) middle_id = wind_speed_ids[Int(length(wind_speed_ids)/2.0)] - + # get direction aep including all wind speeds for that direction - state_aep[i] = calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, + state_aep[i] = calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, ct_model, generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, power_models, rotor_sample_points_y, rotor_sample_points_z, wind_resource, model_set; wind_farm_state_id=middle_id, hours_per_year=hours_per_year, wind_speed_ids=wind_speed_ids) @@ -659,7 +659,7 @@ function calculate_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, state_aep = zeros(arr_type,nstates) Threads.@threads for i = 1:nstates - state_aep[i] = calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, + state_aep[i] = calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, ct_model, generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, power_models, rotor_sample_points_y, rotor_sample_points_z, wind_resource, model_set; wind_farm_state_id=i, hours_per_year=hours_per_year) @@ -667,7 +667,7 @@ function calculate_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, end AEP = sum(state_aep) - + # calculate AEP in serial or in parallel using distributed processing else # if possible, avoid recalculating wakes for more than one speed in each direction @@ -680,17 +680,17 @@ function calculate_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, # take a speed in the middle (so it is not zero) middle_id = wind_speed_ids[max(Int(round(length(wind_speed_ids)/2.0)),1)] - + # get direction aep including all wind speeds for that direction - calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, + calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, ct_model, generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, power_models, rotor_sample_points_y, rotor_sample_points_z, wind_resource, model_set; wind_farm_state_id=middle_id, hours_per_year=hours_per_year, wind_speed_ids=wind_speed_ids) end else AEP = @sync @distributed (+) for i = 1:nstates - - state_aep = calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, + + state_aep = calculate_state_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, ct_model, generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, power_models, rotor_sample_points_y, rotor_sample_points_z, wind_resource, model_set; wind_farm_state_id=i, hours_per_year=hours_per_year) diff --git a/src/thrust_coefficient_models.jl b/src/thrust_coefficient_models.jl index a591997..03a3777 100644 --- a/src/thrust_coefficient_models.jl +++ b/src/thrust_coefficient_models.jl @@ -23,9 +23,9 @@ lowest wind speed to highest wind speed. - `inflow_velocity::Float`: inflow velocity of the wind turbine - `thrust_model::ThrustModelCtPoints`: Struct containing ct and velocity points for ct curve """ -struct ThrustModelCtPoints{ATF} <: AbstractThrustCoefficientModel - vel_points::ATF - ct_points::ATF +struct ThrustModelCtPoints{ATF1,ATF2} <: AbstractThrustCoefficientModel + vel_points::ATF1 + ct_points::ATF2 end """ @@ -76,8 +76,8 @@ end """ _ct_to_axial_ind_func(ct) -Calculate axial induction from the thrust coefficient. See Gebraad et. al. 2017 -"Maximization of the Annual Energy Production of Wind Power Plants by Optimization of +Calculate axial induction from the thrust coefficient. See Gebraad et. al. 2017 +"Maximization of the Annual Energy Production of Wind Power Plants by Optimization of Layout and Yaw-Based Wake Control" # Arguments diff --git a/src/turbines.jl b/src/turbines.jl deleted file mode 100644 index 5b52965..0000000 --- a/src/turbines.jl +++ /dev/null @@ -1,14 +0,0 @@ -abstract type AbstractTurbineDefinition end - -struct TurbineDefinition{TI,AF,CTM,PM} <: AbstractTurbineDefinition - id::TI - rotor_diameter::AF - hub_height::AF - cut_in_speed::AF - rated_speed::AF - cut_out_speed::AF - rated_power::AF - generator_efficiency::AF - ct_model::CTM - power_model::PM -end diff --git a/src/utilities.jl b/src/utilities.jl index 71eeffe..99ed22a 100644 --- a/src/utilities.jl +++ b/src/utilities.jl @@ -1,7 +1,7 @@ """ met2cart(angle_met) -Convert from meteorological polar system (CW, 0 rad.=N, wind from) to cartesian polar system +Convert from meteorological polar system (CW, 0 rad.=N, wind from) to cartesian polar system (CCW, 0 rad.=E, wind to). # Arguments @@ -25,7 +25,7 @@ the positive x. # Arguments - `xlocs::Array`: contains turbine east-west locations in the global reference frame - `ylocs::Array`: contains turbine north-south locations in the global reference frame -- `wind_direction_met::Array`: contains wind direction in radians in meteorological standard +- `wind_direction_met::Array`: contains wind direction in radians in meteorological standard system (N=0 rad, proceeds CW, wind from direction given) """ function rotate_to_wind_direction(xlocs, ylocs, wind_direction_met::Number; center=[0.0,0.0]) @@ -56,7 +56,7 @@ coordinate frame based on the point with the lowest magnitude latitude, - `isnorth::Float`: specifies if the point is in the northern hemisphere (defaul: true) """ function latlong_to_xy(latitude, longitude, utm_zone; isnorth=true) - + # get number of points npoints = length(latitude) @@ -68,7 +68,7 @@ function latlong_to_xy(latitude, longitude, utm_zone; isnorth=true) # get zero point lat long minlla = gd.LLA(latitude[zp], longitude[zp]) - + # get zero point utm minutm = utmregion(minlla) @@ -89,13 +89,13 @@ function latlong_to_xy(latitude, longitude, utm_zone; isnorth=true) y[i] = utm.y - minutm.y end - + return x, y end """ hermite_spline(x, x0, x1, y0, dy0, y1, dy1) - + Produces the y and (optionally) dy values for a hermite cubic spline interpolating between two end points with known slopes @@ -111,12 +111,12 @@ interpolating between two end points with known slopes function hermite_spline(x, x0, x1, y0, dy0, y1, dy1; return_deriv=false) # initialize coefficients for parametric Hermite cubic spline - c3 = (2.0*(y1))/(x0^3 - 3.0*x0^2*x1 + 3.0*x0*x1^2 - x1^3) - - (2.0*(y0))/(x0^3 - 3.0*x0^2*x1 + 3.0*x0*x1^2 - x1^3) + - (dy0)/(x0^2 - 2.0*x0*x1 + x1^2) + + c3 = (2.0*(y1))/(x0^3 - 3.0*x0^2*x1 + 3.0*x0*x1^2 - x1^3) - + (2.0*(y0))/(x0^3 - 3.0*x0^2*x1 + 3.0*x0*x1^2 - x1^3) + + (dy0)/(x0^2 - 2.0*x0*x1 + x1^2) + (dy1)/(x0^2 - 2.0*x0*x1 + x1^2) - c2 = (3.0*(y0)*(x0 + x1))/(x0^3 - 3.0*x0^2*x1 + 3.0*x0*x1^2 - x1^3) - + c2 = (3.0*(y0)*(x0 + x1))/(x0^3 - 3.0*x0^2*x1 + 3.0*x0*x1^2 - x1^3) - ((dy1)*(2.0*x0 + x1))/(x0^2 - 2.0*x0*x1 + x1^2) - ((dy0)*(x0 + 2.0*x1))/(x0^2 - 2.0*x0*x1 + x1^2) - (3.0*(y1)*(x0 + x1))/(x0^3 - 3.0*x0^2*x1 + 3.0*x0*x1^2 - x1^3) @@ -128,7 +128,7 @@ function hermite_spline(x, x0, x1, y0, dy0, y1, dy1; return_deriv=false) c0 = ((y0)*(- x1^3 + 3.0*x0*x1^2))/(x0^3 - 3.0*x0^2*x1 + 3.0*x0*x1^2 - x1^3) - ((y1)*(- x0^3 + 3.0*x1*x0^2))/(x0^3 - 3.0*x0^2*x1 + - 3.0*x0*x1^2 - x1^3) - (x0*x1^2*(dy0))/(x0^2 - 2.0*x0*x1 + x1^2) - + 3.0*x0*x1^2 - x1^3) - (x0*x1^2*(dy0))/(x0^2 - 2.0*x0*x1 + x1^2) - (x0^2*x1*(dy1))/(x0^2 - 2.0*x0*x1 + x1^2) # Solve for y and dy values at the given point @@ -145,7 +145,7 @@ end """ overlap_area_func(turbine_y, turbine_z, rotor_diameter, wake_center_y, wake_center_z, wake_diameter; tol=1E-6) - + Produces the y and (optionally) dy values for a hermite cubic spline interpolating between two end points with known slopes @@ -227,7 +227,7 @@ end Calculate the smooth-max (a.k.a. softmax or LogSumExponential) of the elements in x. -Based on John D. Cook's writings at +Based on John D. Cook's writings at (1) https://www.johndcook.com/blog/2010/01/13/soft-maximum/ and (2) https://www.johndcook.com/blog/2010/01/20/how-to-compute-the-soft-maximum/ @@ -257,7 +257,7 @@ end Calculate the smoothmax (a.k.a. softmax or LogSumExponential) of the elements in x. -Based on John D. Cook's writings at +Based on John D. Cook's writings at (1) https://www.johndcook.com/blog/2010/01/13/soft-maximum/ and (2) https://www.johndcook.com/blog/2010/01/20/how-to-compute-the-soft-maximum/ @@ -272,7 +272,7 @@ And based on article in FeedlyBlog function smooth_max(x; s=10.0) # non-overflowing version of Smooth Max function (see ref 2 and 3 above) - + # get the maximum value and the index of maximum value max_val, max_ind = findmax(real(x)) @@ -311,7 +311,7 @@ function closeBndryLists(region_bndry_x, region_bndry_y) # Append the initial points to the end of that row (if needed) region_bndry_x[i], region_bndry_y[i] = closeBndryList(region_bndry_x[i], region_bndry_y[i]) end - + return region_bndry_x, region_bndry_y end @@ -382,21 +382,21 @@ function calcMinorAngle(bndry_x, bndry_y, bndry_z=[0,0,0]) ABx = bndry_x[2]-bndry_x[1] ABy = bndry_y[2]-bndry_y[1] ABz = bndry_z[2]-bndry_z[1] - + BCx = bndry_x[2]-bndry_x[3] BCy = bndry_y[2]-bndry_y[3] BCz = bndry_z[2]-bndry_z[3] - + Num = (ABx*BCx) + (ABy*BCy) + (ABz*BCz) # Dot Product - + Denom = norm([ABx, ABy, ABz]) * norm([BCx, BCy, BCz]) # Multiplication of magnitudes Theta = acosd(Num/Denom) # Get the angle formed - # If it's greater than 180, get the + # If it's greater than 180, get the if (Theta > 180.0) Theta = 360.0 - Theta # Get the explementary angle end - + return Theta end @@ -421,7 +421,7 @@ function calcSmallestAngle(bndry_x_clsd, bndry_y_clsd) bndryPts_x_loopd = vcat(bndry_x_clsd, bndry_x_clsd[2]) bndryPts_y_loopd = vcat(bndry_y_clsd, bndry_y_clsd[2]) end - + #- Calculate the smallest angle -# smallest_angle = 360 for i in 1:num_angles @@ -431,7 +431,7 @@ function calcSmallestAngle(bndry_x_clsd, bndry_y_clsd) smallest_angle = temp_angle end end - + return smallest_angle end @@ -454,7 +454,7 @@ function getPerimeterLength(bndry_x_clsd, bndry_y_clsd) for i in 1:num_bndry_pts nLength[i] = norm([bndry_x_clsd[i+1]-bndry_x_clsd[i], bndry_y_clsd[i+1]-bndry_y_clsd[i]]) end - + return nLength end @@ -480,7 +480,7 @@ end single_boundary_normals_calculator(boundary_vertices) -Outputs the unit vectors perpendicular to each edge of a polygon, given the Cartesian +Outputs the unit vectors perpendicular to each edge of a polygon, given the Cartesian coordinates for the polygon's vertices. # Arguments @@ -504,10 +504,10 @@ function single_boundary_normals_calculator(boundary_vertices) # create a vector normal to the boundary boundary_normals[i, :] = [ -(boundary_vertices[i+1, 2] - boundary_vertices[i, 2]) ; boundary_vertices[i+1, 1] - boundary_vertices[i, 1] ] - + # normalize the vector boundary_normals[i, :] = boundary_normals[i, :] / norm(boundary_normals[i, :]) - + end return boundary_normals @@ -518,7 +518,7 @@ end boundary_normals_calculator(boundary_vertices; nboundaries=1) -Outputs the unit vectors perpendicular to each edge of each polygon in a set of polygons, +Outputs the unit vectors perpendicular to each edge of each polygon in a set of polygons, given the Cartesian coordinates for each polygon's vertices. # Arguments @@ -526,7 +526,7 @@ given the Cartesian coordinates for each polygon's vertices. - `nboundaries::Int` : the number of boundaries in the set """ function boundary_normals_calculator(boundary_vertices; nboundaries=1) - + if nboundaries == 1 boundary_normals = single_boundary_normals_calculator(boundary_vertices) else @@ -542,13 +542,13 @@ end _remove_perimeter_points!(n; alpha=0.0) -Internal function. Removes points outside or outside and on the border of the rotor-swept - area +Internal function. Removes points outside or outside and on the border of the rotor-swept + area # Arguments - `y::AbstractArray`: horizontal point locations -- `z::AbstractArray`: vertical point locations -- `use_perimeter_points::Bool`: flag that determines whether or not to include points on the +- `z::AbstractArray`: vertical point locations +- `use_perimeter_points::Bool`: flag that determines whether or not to include points on the boundary of the rotor-swept area """ function _remove_out_of_bounds_points(y, z, use_perimeter_points) @@ -571,7 +571,7 @@ end sunflower_points(n; alpha=0.0) -Generates points in a circle of radius=1 using the sunflower packing algorithm. +Generates points in a circle of radius=1 using the sunflower packing algorithm. # Arguments - `n::Float`: number of points to generate @@ -615,7 +615,7 @@ end grid_points(n) -Generates points in a grid. If n is not a perfect square, then the nearest square root will +Generates points in a grid. If n is not a perfect square, then the nearest square root will be used for the side length of the grid. # Arguments @@ -628,10 +628,10 @@ function grid_points(n) # determine length of x and y in grid, round if not perfect square sidepoints = Int(round(sqrt(n), digits=0)) - # generate horizontal points + # generate horizontal points y = -1.0:2.0/(sidepoints-1.0):1.0 - # generate vertical points + # generate vertical points z = -1.0:2.0/(sidepoints-1.0):1.0 # generate grid @@ -656,24 +656,24 @@ using the sunflower packcing algorithm. - `nsamplepoints::Int`: controls how many sample points to generate - `alpha::Float`: Controls smoothness of the sunflower algorithm boundary. alpha=0 is the standard "jagged edge" sunflower algoirthm and alpha=1 results in a smooth boundary. -- `pradius::Float`: the percent of the rotor radius to use in generating initial point grid -- `use_perimeter_points`: whether or not to include point exactly on the perimeter of the - rotor swept area +- `pradius::Float`: the percent of the rotor radius to use in generating initial point grid +- `use_perimeter_points`: whether or not to include point exactly on the perimeter of the + rotor swept area """ function rotor_sample_points(nsamplepoints=1; method="sunflower", alpha=0.0, use_perimeter_points=true, pradius=1.0) if nsamplepoints > 1 if method == "sunflower" - rotor_sample_points_y, rotor_sample_points_z = ff.sunflower_points(nsamplepoints, alpha=alpha) + rotor_sample_points_y, rotor_sample_points_z = sunflower_points(nsamplepoints, alpha=alpha) elseif method == "grid" - rotor_sample_points_y, rotor_sample_points_z = ff.grid_points(nsamplepoints) + rotor_sample_points_y, rotor_sample_points_z = grid_points(nsamplepoints) end - # adjust to desired radius + # adjust to desired radius rotor_sample_points_y .*= pradius rotor_sample_points_z .*= pradius - # remove any points outside the swept area + # remove any points outside the swept area rotor_sample_points_y, rotor_sample_points_z = _remove_out_of_bounds_points(rotor_sample_points_y, rotor_sample_points_z, use_perimeter_points) else @@ -683,70 +683,70 @@ function rotor_sample_points(nsamplepoints=1; method="sunflower", alpha=0.0, use return rotor_sample_points_y, rotor_sample_points_z end -""" - print_state_layouts_in_cartesian_frame(turbinex, turbiney, winddirections) +# """ +# print_state_layouts_in_cartesian_frame(turbinex, turbiney, winddirections) -Given a wind farm layout in the global reference frame, print the layout rotated to the -cartesian frame with wind to the positive x axis (right) for all wind directions. +# Given a wind farm layout in the global reference frame, print the layout rotated to the +# cartesian frame with wind to the positive x axis (right) for all wind directions. -# Arguments -- `turbinex::Array{T,1}`: x locations of turbines in global reference frame -- `turbiney::Array{T,1}`: y locations of turbines in global reference frame -- `winddirections::Array{T,1}`: all wind directions in radians in meteorological coordinates (0 rad. = from North) -""" -function print_layout_in_cartesian_frame_excel(turbinex, turbiney, rotordiameter, winddirections, outfile; center=[0.0,0.0], plot_layouts=false, round_locations=false) - # get number of directions - ndirections = length(winddirections) - - # initialize excel file - XLSX.openxlsx(outfile, mode="w") do xf - - sheet = xf[1] - sheet["A1"] = "wind direction" - sheet["B1"] = "turbine x" - sheet["C1"] = "turbine y" - - # loop through directions - for i in 1:ndirections - - # rotate turbinex and turbiney to current direction - rotx, roty = rotate_to_wind_direction(turbinex, turbiney, winddirections[i], center=center) - - # round if desired - if round_locations - rotx = round.(rotx) - roty = round.(roty) - end +# # Arguments +# - `turbinex::Array{T,1}`: x locations of turbines in global reference frame +# - `turbiney::Array{T,1}`: y locations of turbines in global reference frame +# - `winddirections::Array{T,1}`: all wind directions in radians in meteorological coordinates (0 rad. = from North) +# """ +# function print_layout_in_cartesian_frame_excel(turbinex, turbiney, rotordiameter, winddirections, outfile; center=[0.0,0.0], plot_layouts=false, round_locations=false) +# # get number of directions +# ndirections = length(winddirections) - # plot layouts if desired, pause for 5 seconds on each - if plot_layouts - fig, ax = plt.subplots() - ax.set(aspect="equal", xlim=[minimum(turbinex)-200,maximum(turbinex)+200], ylim=[minimum(turbiney)-200,maximum(turbiney)+200]) - ff.plotlayout!(ax, rotx, roty, rotordiameter) - ff.plotlayout!(ax, [rotx[1], rotx[20]], [roty[1], roty[20]], rotordiameter, fill=true, color="r") - plt.title("$(winddirections[i]*180.0/pi)") - plt.show() +# # initialize excel file +# XLSX.openxlsx(outfile, mode="w") do xf - # sleep(1) +# sheet = xf[1] +# sheet["A1"] = "wind direction" +# sheet["B1"] = "turbine x" +# sheet["C1"] = "turbine y" - end +# # loop through directions +# for i in 1:ndirections - XLSX.rename!(sheet, "rotated layouts") +# # rotate turbinex and turbiney to current direction +# rotx, roty = rotate_to_wind_direction(turbinex, turbiney, winddirections[i], center=center) - # print rotated coordinates to specified output - sheet["A$(i+1)"] = winddirections[i]*180.0/pi - sheet["B$(i+1)"] = join(rotx, ",") - sheet["C$(i+1)"] = join(roty, ",") +# # round if desired +# if round_locations +# rotx = round.(rotx) +# roty = round.(roty) +# end - end - end -end +# # plot layouts if desired, pause for 5 seconds on each +# if plot_layouts +# fig, ax = plt.subplots() +# ax.set(aspect="equal", xlim=[minimum(turbinex)-200,maximum(turbinex)+200], ylim=[minimum(turbiney)-200,maximum(turbiney)+200]) +# plotlayout!(ax, rotx, roty, rotordiameter) +# plotlayout!(ax, [rotx[1], rotx[20]], [roty[1], roty[20]], rotordiameter, fill=true, color="r") +# plt.title("$(winddirections[i]*180.0/pi)") +# plt.show() + +# # sleep(1) + +# end + +# XLSX.rename!(sheet, "rotated layouts") + +# # print rotated coordinates to specified output +# sheet["A$(i+1)"] = winddirections[i]*180.0/pi +# sheet["B$(i+1)"] = join(rotx, ",") +# sheet["C$(i+1)"] = join(roty, ",") + +# end +# end +# end function wrap_180(x) """ Wrap an angle to between -180 and 180 adapted from NREL's floris - """ + """ x[x .<= -180.0] .+= 360.0 x[x .> 180.0] .-= 360.0 @@ -754,7 +754,7 @@ function wrap_180(x) return(x) end -# the following is from floris for calcculating wake counts +# the following is from floris for calculating wake counts """ wake_count_iec(turbinex, turbiney, winddirection, diameter; return_turbines=true) @@ -765,19 +765,19 @@ end in Figure A.1 in Annex A of the IEC 61400-12-1:2017 standard. # Arguments -- `turbinex::Array{T,1}`: x locations of turbines in global reference frame +- `turbinex::Array{T,1}`: x locations of turbines in global reference frame - `turbiney::Array{T,1}`: y locations of turbines in global reference frame - `winddirection::Float`: wind direction in radians in meteorological coordinates (0 rad. = from North) - `diameter::Array{T,1}`: diameters of all wind turbines """ function wake_count_iec(turbinex, turbiney, wd::Real, diameter; return_turbines=true) - # convert wind direction to degrees - wd *= 180.0/pi + # convert wind direction to degrees + wd *= 180.0/pi # get number of turbines nturbines = length(turbinex) - + # set indices of all turbines turbines = collect(1:nturbines) @@ -792,10 +792,10 @@ function wake_count_iec(turbinex, turbiney, wd::Real, diameter; return_turbines= # calculate distance in diameters from turbi to all other turbines dists = hypot.(turbinex[other_turbines] .- turbinex[turbi], turbiney[other_turbines] .- turbiney[turbi])./diameter[other_turbines] - - # calculate angles in degrees from other turbines to turbi + + # calculate angles in degrees from other turbines to turbi angles = rad2deg.(atan.(turbinex[other_turbines] .- turbinex[turbi], turbiney[other_turbines] .- turbiney[turbi])) - + waked = dists .<= 2.0 waked = waked .| (&).( (dists .<= 20.0), @@ -813,10 +813,10 @@ end function wake_count_iec(turbinex, turbiney, wd::AbstractArray, diameter; return_turbines=true) - # initialize wake list + # initialize wake list wake_list = zeros(Int, (length(wd), length(turbinex))) - # call wake_count_iec for each direction + # call wake_count_iec for each direction for i = 1:length(wd) wake_list[i,:] = wake_count_iec(turbinex, turbiney, wd[i], diameter) end @@ -827,18 +827,18 @@ end """ find_upstream_turbines(turbinex, turbiney, winddirection, diameter; inverse=false) -A convenience function to quickly find either which turbines are waked, or those that are -not. +A convenience function to quickly find either which turbines are waked, or those that are +not. # Arguments -- `turbinex::Array{T,1}`: x locations of turbines in global reference frame +- `turbinex::Array{T,1}`: x locations of turbines in global reference frame - `turbiney::Array{T,1}`: y locations of turbines in global reference frame - `winddirection::Real` or `winddirection::AbstractArray`: wind direction in radians in meteorological coordinates (0 rad. = from North) - `diameter::Array{T,1}`: diameters of all wind turbines """ function find_upstream_turbines(turbinex, turbiney, winddirection::AbstractArray, diameter; inverse=false) - # find wake count for all turbines in given wind direction + # find wake count for all turbines in given wind direction wake_count = [] for wd in winddirection @@ -853,7 +853,7 @@ function find_upstream_turbines(turbinex, turbiney, winddirection::AbstractArray end return returnarray else - # return unwaked turbines + # return unwaked turbines returnarray = [] for i = 1:length(winddirection) push!(returnarray, collect(1:length(turbinex))[wake_count[i] .== 0]) @@ -865,14 +865,14 @@ end function find_upstream_turbines(turbinex, turbiney, winddirection::Real, diameter; inverse=false) - # find wake count for all turbines in given wind direction + # find wake count for all turbines in given wind direction wake_count = wake_count_iec(turbinex, turbiney, winddirection, diameter) if inverse # return waked turbines return collect(1:length(turbinex))[wake_count .!= 0] else - # return unwaked turbines + # return unwaked turbines return collect(1:length(turbinex))[wake_count .== 0] end @@ -922,11 +922,11 @@ end """ round_farm_random_start(rotor_diameter, center, radius; min_spacing=2., min_spacing_random=3., method="individual") - + Generates starting locations for multi-start optimization approaches when the farm boundary is round. # Arguments -- `rotor_diameter::Number`: wind turbine diameter +- `rotor_diameter::Number`: wind turbine diameter - `center::Number`: wind farm center - `radius::Number`: wind farm radius - `diameter::Array{T,1}`: diameters of all wind turbines @@ -990,35 +990,35 @@ function round_farm_random_start(rotor_diameter, center, radius; nturbines=nothi end elseif method == "angle" turbinex, turbiney = round_farm_concentric_start(copy(rotor_diameter), copy(center*rotor_diameter), copy(radius*rotor_diameter), min_spacing=min_spacing_random) - + turbinex /= rotor_diameter turbiney /= rotor_diameter turbinex .-= center[1] turbiney .-= center[2] - # get rotation angle + # get rotation angle step = 0.001 rotationangle = rand(0:step:(2*pi-step)) # in radians # rotate turbinex[:], turbiney[:] = rotate_to_wind_direction(turbinex, turbiney, rotationangle) - # translate - turbinex .+= center[1] - turbiney .+= center[2] + # translate + turbinex .+= center[1] + turbiney .+= center[2] elseif method == "angle-each-circle" turbinex, turbiney = round_farm_concentric_start(copy(rotor_diameter), copy(center*rotor_diameter), copy(radius*rotor_diameter), min_spacing=min_spacing_random) - + turbinex /= rotor_diameter turbiney /= rotor_diameter turbinex .-= center[1] turbiney .-= center[2] - # get rotation angle + # get rotation angle step = 0.001 circleidx = [2:7, 8:19, 20:38] for i in 1:3 @@ -1027,15 +1027,15 @@ function round_farm_random_start(rotor_diameter, center, radius; nturbines=nothi # rotate circle i turbinex[circleidx[i]], turbiney[circleidx[i]] = rotate_to_wind_direction(turbinex[circleidx[i]], turbiney[circleidx[i]], rotationangle) end - # translate - turbinex .+= center[1] - turbiney .+= center[2] + # translate + turbinex .+= center[1] + turbiney .+= center[2] elseif method == "concentric" # calculate how many circles can be fit in the wind farm area maxcircles = floor(radius / min_spacing_random) - # get max number of turbines that will fit on the boundary + # get max number of turbines that will fit on the boundary alpha_min = 2.0 * asin.(min_spacing_random / (2.0 * radius)) max_boundary_turbines = floor.(2.0 * pi / alpha_min) @@ -1044,18 +1044,18 @@ function round_farm_random_start(rotor_diameter, center, radius; nturbines=nothi else mincircles = 2 end - mincircles = 3 + mincircles = 3 # choose how many circles (random) ncircles = Int(rand(mincircles:maxcircles)) # initialize circles radii = range((radius/ncircles), radius, length = Int(ncircles)) - + remaining_turbines = nturbines - 1 circle_turbines = zeros(ncircles) for i = 1:length(radii)-1 - # get max number of turbines that will fit on the circle + # get max number of turbines that will fit on the circle alpha_min = 2.0 * asin.(min_spacing_random / (2.0 * radii[i])) max_circle_turbines = minimum([floor.(2.0 * pi / alpha_min), remaining_turbines*0.5]) @@ -1064,7 +1064,7 @@ function round_farm_random_start(rotor_diameter, center, radius; nturbines=nothi # select how many turbines should be placed on the circle circle_turbines[i] = rand(min_circle_turbines:max_circle_turbines) - # update remaining turbines + # update remaining turbines remaining_turbines -= circle_turbines[i] end @@ -1089,39 +1089,39 @@ function round_farm_random_start(rotor_diameter, center, radius; nturbines=nothi end end - # get rotation angle + # get rotation angle step = 0.001 rotationangle = rand(0:step:(2*pi-step)) # in radians - + # rotate turbinex[:], turbiney[:] = rotate_to_wind_direction(turbinex, turbiney, rotationangle) - # translate - turbinex .+= center[1] - turbiney .+= center[2] + # translate + turbinex .+= center[1] + turbiney .+= center[2] elseif method == "grid" elseif method == "vrsunflower" - # puting random number of turbines on the boundary, distribute the rest using sunflower + # puting random number of turbines on the boundary, distribute the rest using sunflower - # set minimum number of turbines for the boundary + # set minimum number of turbines for the boundary min_boundary_turbines = floor(0.3*nturbines) - # get max number of turbines that will fit on the boundary + # get max number of turbines that will fit on the boundary alpha_min = 2.0 * asin.(min_spacing_random / (2.0 * radius)) max_boundary_turbines = floor.(2.0 * pi / alpha_min) if max_boundary_turbines > floor(0.6*nturbines) max_boundary_turbines = 0.6*nturbines end - # get number of turbines to put on the boundary + # get number of turbines to put on the boundary nbturbines = Int(rand(min_boundary_turbines:max_boundary_turbines)) - # get angle for chosen number of boundary turbines + # get angle for chosen number of boundary turbines alpha_b = 2*pi/nbturbines - # place boundary turbines + # place boundary turbines for turb in 1:nbturbines angle = alpha_b*(turb-1) w = radius*cos(angle) @@ -1130,21 +1130,21 @@ function round_farm_random_start(rotor_diameter, center, radius; nturbines=nothi turbiney[Int(turb)] = h end - # get number of interior turbines - niturbines = nturbines - nbturbines + # get number of interior turbines + niturbines = nturbines - nbturbines - # place interior turbines + # place interior turbines xi, yi = rotor_sample_points(niturbines, alpha=1).*(radius - min_spacing_random) if niturbines > 0 - turbinex[nbturbines+1:end] = xi - turbiney[nbturbines+1:end] = yi + turbinex[nbturbines+1:end] = xi + turbiney[nbturbines+1:end] = yi end - # check spacing + # check spacing for i = 1:nturbines for j = i+1:nturbines dist = hypot(turbinex[i]-turbinex[j], turbiney[i]-turbiney[j]) - + if dist < min_spacing_random # println(dist) throw(ErrorException("too many turbines to use $method in given space")) @@ -1152,26 +1152,26 @@ function round_farm_random_start(rotor_diameter, center, radius; nturbines=nothi end end - # get rotation angle + # get rotation angle step = 0.001 rotationangle = rand(0:step:(2*pi-step)) # in radians - + # rotate turbinex[:], turbiney[:] = rotate_to_wind_direction(turbinex, turbiney, rotationangle) - # translate - turbinex .+= center[1] - turbiney .+= center[2] + # translate + turbinex .+= center[1] + turbiney .+= center[2] - elseif method == "sunflower" - # get locations + elseif method == "sunflower" + # get locations x, y = rotor_sample_points(38, alpha=1.0).*radius - # check spacing + # check spacing for i = 1:nturbines for j = i+1:nturbines dist = hypot(x[i]-x[j], y[i]-y[j]) - + if dist < min_spacing_random # println(dist) throw(ErrorException("too many turbines to use sunflower in given space")) @@ -1179,14 +1179,14 @@ function round_farm_random_start(rotor_diameter, center, radius; nturbines=nothi end end - # translate - x .+= center[1] - y .+= center[2] + # translate + x .+= center[1] + y .+= center[2] - # get rotation angle + # get rotation angle step = 0.001 rotationangle = rand(0:step:(2*pi-step)) # in radians - + # rotate turbinex[:], turbiney[:] = rotate_to_wind_direction(x, y, rotationangle) else @@ -1197,66 +1197,66 @@ function round_farm_random_start(rotor_diameter, center, radius; nturbines=nothi return turbinex.*rotor_diameter, turbiney.*rotor_diameter end -function generate_round_layouts(nlayouts, rotor_diameter; farm_center=0., farm_diameter=4000., base_spacing=5., min_spacing=1., - output_directory=nothing, show=false, save_layouts=false, startingindex=1, method="individual") - - if nlayouts > 10 && show == true - error("do you really want to see $nlayouts plots in serial?") - end - - boundary_radius = 0.5*(farm_diameter - rotor_diameter) - # println(boundary_radius) - area = pi * boundary_radius^2 - - turbinex, turbiney = round_farm_concentric_start(copy(rotor_diameter), copy(farm_center), copy(boundary_radius), min_spacing=copy(base_spacing)) - - nturbines = size(turbinex)[1] - # println("nturbines: $nturbines") - effective_rows = sqrt(nturbines) - effective_row_length = sqrt(area) - effective_spacing = effective_row_length / (effective_rows - 1.) - l = 1 - - if save_layouts - df = DataFrame(turbinex=turbinex./rotor_diameter, turbiney=turbiney./rotor_diameter) - CSV.write(output_directory*"nTurbs$(nturbines)_spacing$(base_spacing)_layout_$(l+startingindex-1).txt", - df, header=["turbineX", "turbineY"]) - end - if show - fig, ax = subplots(1) - plotlayout!(ax, turbinex, turbiney, ones(nturbines).*rotor_diameter) - plt.show() - end - - if nlayouts > 1 - for l in 2:nlayouts - println("Generating Layout $l") - - turbinex, turbiney = round_farm_random_start(copy(rotor_diameter), copy(farm_center), - copy(boundary_radius), min_spacing=copy(base_spacing), - min_spacing_random=min_spacing, method=method, nturbines=nturbines) - - if save_layouts - df = DataFrame(turbinex=turbinex./rotor_diameter, turbiney=turbiney./rotor_diameter) - CSV.write(output_directory*"nTurbs$(nturbines)_spacing$(min_spacing)_layout_$(l+startingindex-1).txt", - df, header=["turbineX", "turbineY"]) - end - if show - fig, ax = plt.subplots(1) - plotlayout!(ax, turbinex, turbiney, ones(nturbines).*rotor_diameter, xlim=[-boundary_radius-rotor_diameter, boundary_radius+rotor_diameter], ylim=[-boundary_radius-rotor_diameter, boundary_radius+rotor_diameter]) - circle = matplotlib.patches.Circle(farm_center, boundary_radius, fill=false, color="r") - ax.add_patch(circle) - plt.show() - end - end - end -end +# function generate_round_layouts(nlayouts, rotor_diameter; farm_center=0., farm_diameter=4000., base_spacing=5., min_spacing=1., +# output_directory=nothing, show=false, save_layouts=false, startingindex=1, method="individual") + +# if nlayouts > 10 && show == true +# error("do you really want to see $nlayouts plots in serial?") +# end + +# boundary_radius = 0.5*(farm_diameter - rotor_diameter) +# # println(boundary_radius) +# area = pi * boundary_radius^2 + +# turbinex, turbiney = round_farm_concentric_start(copy(rotor_diameter), copy(farm_center), copy(boundary_radius), min_spacing=copy(base_spacing)) + +# nturbines = size(turbinex)[1] +# # println("nturbines: $nturbines") +# effective_rows = sqrt(nturbines) +# effective_row_length = sqrt(area) +# effective_spacing = effective_row_length / (effective_rows - 1.) +# l = 1 + +# if save_layouts +# df = DataFrame(turbinex=turbinex./rotor_diameter, turbiney=turbiney./rotor_diameter) +# CSV.write(output_directory*"nTurbs$(nturbines)_spacing$(base_spacing)_layout_$(l+startingindex-1).txt", +# df, header=["turbineX", "turbineY"]) +# end +# if show +# fig, ax = subplots(1) +# plotlayout!(ax, turbinex, turbiney, ones(nturbines).*rotor_diameter) +# plt.show() +# end + +# if nlayouts > 1 +# for l in 2:nlayouts +# println("Generating Layout $l") + +# turbinex, turbiney = round_farm_random_start(copy(rotor_diameter), copy(farm_center), +# copy(boundary_radius), min_spacing=copy(base_spacing), +# min_spacing_random=min_spacing, method=method, nturbines=nturbines) + +# if save_layouts +# df = DataFrame(turbinex=turbinex./rotor_diameter, turbiney=turbiney./rotor_diameter) +# CSV.write(output_directory*"nTurbs$(nturbines)_spacing$(min_spacing)_layout_$(l+startingindex-1).txt", +# df, header=["turbineX", "turbineY"]) +# end +# if show +# fig, ax = plt.subplots(1) +# plotlayout!(ax, turbinex, turbiney, ones(nturbines).*rotor_diameter, xlim=[-boundary_radius-rotor_diameter, boundary_radius+rotor_diameter], ylim=[-boundary_radius-rotor_diameter, boundary_radius+rotor_diameter]) +# circle = matplotlib.patches.Circle(farm_center, boundary_radius, fill=false, color="r") +# ax.add_patch(circle) +# plt.show() +# end +# end +# end +# end """ pointonline(p, v1, v2; tol=1E-6) Given a line determined two points (v1 and v2) determine if the point (p) lies on the line -between those points. +between those points. Returns true if the point lies on the line (within the given tolerance), false otherwise. @@ -1280,11 +1280,11 @@ end """ pointinpolygon(point, vertices, normals=nothing; s=700, method="raycasting", shift=1E-10, return_distance=true) -Given a polygon determined by a set of vertices, determine the signed distance from the point -to the polygon. +Given a polygon determined by a set of vertices, determine the signed distance from the point +to the polygon. -Returns the negative (-) distance if the point is inside or on the polygon, positive (+) -otherwise. If return_distance is set to false, then returns -1 if in polygon or on the +Returns the negative (-) distance if the point is inside or on the polygon, positive (+) +otherwise. If return_distance is set to false, then returns -1 if in polygon or on the boundary, and 1 otherwise. # Arguments @@ -1302,7 +1302,7 @@ function pointinpolygon(point, vertices, normals=nothing; s=700, method="raycast throw(ArgumentError("point coordinates may not be given as Ints, must use Floats of some kind. point used $(typeof(point[1]))")) end - if normals === nothing + if normals === nothing normals = boundary_normals_calculator(vertices) end @@ -1331,7 +1331,7 @@ function pointinpolygon(point, vertices, normals=nothing; s=700, method="raycast onvertex = false onedge = false - # initial distance value + # initial distance value c = 0.0 # make sure that the point is not exactly on a vertex @@ -1346,50 +1346,50 @@ function pointinpolygon(point, vertices, normals=nothing; s=700, method="raycast break # # if the point is approximately on a vertex or face, move the point slightly # # this introduces some slight error, but should be well within the error - # # for actual turbine placement. - - # # The direction moved is perpendicular to line between the previous and + # # for actual turbine placement. + + # # The direction moved is perpendicular to line between the previous and # # following vertices to avoid moving along an adjacent face # if i == 1 # pre_direction_vector = vertices[i+1,:] - vertices[nvertices, :] # else # pre_direction_vector = vertices[i+1,:] - vertices[i-1, :] # end - + # # get a vector perpendicular to the pre_direction_vector # perpendicular_direction = [pre_direction_vector[2], -pre_direction_vector[1]] # # normalize perpendicular vector to make it a unit vector # # perpendicular_direction ./= norm(perpendicular_direction) # perpendicular_direction ./= nansafesqrt(sum(perpendicular_direction.^2)) - + # # move the point by shift in the direction of the perpendicular vector # point .+= shift*perpendicular_direction # break - + elseif pointonline(point, vertices[i,:], vertices[i+1,:], tol=shift/2.0) onedge = true break # # if the point is approximately on a vertex or face, move the point slightly # # this introduces some slight error, but should be well within the error - # # for actual turbine placement. - - # # The direction moved is perpendicular to line between the previous and + # # for actual turbine placement. + + # # The direction moved is perpendicular to line between the previous and # # following vertices to avoid moving along an adjacent face # if i == 1 # pre_direction_vector = vertices[i+1,:] - vertices[nvertices, :] # else # pre_direction_vector = vertices[i+1,:] - vertices[i-1, :] # end - + # # get a vector perpendicular to the pre_direction_vector # perpendicular_direction = [pre_direction_vector[2], -pre_direction_vector[1]] # # normalize perpendicular vector to make it a unit vector # # perpendicular_direction ./= norm(perpendicular_direction) # perpendicular_direction ./= nansafesqrt(sum(perpendicular_direction.^2)) - + # # move the point by shift in the direction of the perpendicular vector # point .+= shift*perpendicular_direction @@ -1405,21 +1405,21 @@ function pointinpolygon(point, vertices, normals=nothing; s=700, method="raycast # check and handle if vertex is on an edge # for i = 1:nvertices - + # onedge = pointonline(point, vertices[i,:], vertices[i+1,:], tol=shift/2.0) # # break # if onedge || onvertex - # # # get a vector for the edge - # # vectoredge = vertices[i+1,:] - vertices[i,:] + # # # get a vector for the edge + # # vectoredge = vertices[i+1,:] - vertices[i,:] - # # # get a vector perpendicular to the edge + # # # get a vector perpendicular to the edge # # vectorperpendicular = [vectoredge[2], -vectoredge[1]] - # # # get a unit vector perpendicular to the edge + # # # get a unit vector perpendicular to the edge # # vectorperpendicularhat = vectorperpendicular./nansafenorm(vectorperpendicular) - # # # get distance from edge to point + # # # get distance from edge to point # # c = -abs_smooth(dot(point, vectorperpendicularhat), eps()) # # break @@ -1432,23 +1432,23 @@ function pointinpolygon(point, vertices, normals=nothing; s=700, method="raycast # # end # # if the point is approximately on a vertex or face, move the point slightly # # this introduces some slight error, but should be well within the error - # # for actual turbine placement. - - # # The direction moved is perpendicular to line between the previous and + # # for actual turbine placement. + + # # The direction moved is perpendicular to line between the previous and # # following vertices to avoid moving along an adjacent face # if i == 1 # pre_direction_vector = vertices[i+1,:] - vertices[nvertices, :] # else # pre_direction_vector = vertices[i+1,:] - vertices[i-1, :] # end - + # # get a vector perpendicular to the pre_direction_vector # perpendicular_direction = [pre_direction_vector[2], -pre_direction_vector[1]] # # normalize perpendicular vector to make it a unit vector # # perpendicular_direction ./= norm(perpendicular_direction) # perpendicular_direction ./= nansafesqrt(sum(perpendicular_direction.^2)) - + # # move the point by shift in the direction of the perpendicular vector # point .+= shift*perpendicular_direction # end @@ -1456,10 +1456,10 @@ function pointinpolygon(point, vertices, normals=nothing; s=700, method="raycast # iterate through each boundary for j = 1:nvertices - + # check if x-coordinate of turbine is between the x-coordinates of the two boundary vertices if real(vertices[j, 1]) <= real(point[1]) < real(vertices[j+1, 1]) || real(vertices[j, 1]) >= real(point[1]) > real(vertices[j+1, 1]) - + # check to see if the turbine is below the boundary y = (vertices[j+1, 2] - vertices[j, 2]) / (vertices[j+1, 1] - vertices[j, 1]) * (point[1] - vertices[j, 1]) + vertices[j, 2] if real(point[2]) < real(y) #(vertices[j+1, 2] - vertices[j, 2]) / (vertices[j+1, 1] - vertices[j, 1]) * (point[1] - vertices[j, 1]) + vertices[j, 2] @@ -1498,22 +1498,22 @@ function pointinpolygon(point, vertices, normals=nothing; s=700, method="raycast # else # # the vertical ray intersects the boundary # intersection_counter += 1 - # end + # end end - + if return_distance #&& !onedge # define the vector from the turbine to the second point of the face turbine_to_second_facepoint = vertices[j+1, :] - point # dy/dp = -1 - + # find perpendicular distance from turbine to current face (vector projection) - + # get vector defining the boundary boundary_vector = vertices[j+1, :] - vertices[j, :] - + # check if perpendicular distance is the shortest if real(dot(boundary_vector, -turbine_to_first_facepoint)) > 0 && real(dot(boundary_vector,turbine_to_second_facepoint)) > 0 # if boundary_vector <= turbine_to_first_facepoint && boundary_vector <= turbine_to_second_facepoint - + # perpendicular distance from turbine to face d = dot(turbine_to_first_facepoint, normals[j,:]) if !onedge && !onvertex @@ -1524,34 +1524,34 @@ function pointinpolygon(point, vertices, normals=nothing; s=700, method="raycast # check if distance to first facepoint is shortest elseif real(dot(boundary_vector, -turbine_to_first_facepoint)) < 0 - + # distance from turbine to first facepoint # turbine_to_face_distance[j] = norm(turbine_to_first_facepoint) turbine_to_face_distance[j] = nansafenorm(turbine_to_first_facepoint) - + # distance to second facepoint is shortest else - + # distance from turbine to second facepoint # turbine_to_face_distance[j] = norm(turbine_to_second_facepoint) turbine_to_face_distance[j] = nansafenorm(turbine_to_second_facepoint) - + end - + # reset for next face iteration turbine_to_first_facepoint = turbine_to_second_facepoint # dy/dx = 1 # (for efficiency, so we don't have to recalculate for the same vertex twice) end end - + if return_distance # magnitude of the constraint value # if !onedge - # c = -ff.smooth_max(-turbine_to_face_distance, s=s) + # c = -smooth_max(-turbine_to_face_distance, s=s) # # c = -ksmax(-turbine_to_face_distance, s) # end - c = -ff.smooth_max(-turbine_to_face_distance, s=s) + c = -smooth_max(-turbine_to_face_distance, s=s) # c = -ksmax(-turbine_to_face_distance, s) - + # sign of the constraint value (- is inside, + is outside) if mod(intersection_counter, 2) == 1 || onvertex || onedge c = -c @@ -1570,7 +1570,7 @@ end """ nansafesqrt(a) -Calculate the square root of a number, but if the number is less than the given tolerance +Calculate the square root of a number, but if the number is less than the given tolerance then use the line y = a(sqrt(eps())/eps()) so that the derivative is well defined. # Arguments @@ -1589,15 +1589,15 @@ end """ nansafenorm(v) -Calculate the norm of a vector, but if the sum of the squares is less than the given tolerance +Calculate the norm of a vector, but if the sum of the squares is less than the given tolerance then use the line y = a(sqrt(eps())/eps()) so that the derivative is well defined. # Arguments -- `v::Vector{}`: takes the norm of this vector, but avoids NaN by using a linear +- `v::Vector{}`: takes the norm of this vector, but avoids NaN by using a linear approximation of sqrt near 0. """ function nansafenorm(v::Vector) - + return nansafesqrt(sum(v.^2)) end @@ -1605,7 +1605,7 @@ end """ star_boundary(n) -Generate the points for a star with n points +Generate the points for a star with n points # Arguments - `n::Int`: The number of points the star should have @@ -1616,30 +1616,30 @@ Generate the points for a star with n points function star_boundary(n, ri, ro, rotation=0.0) - # outer angles + # outer angles ao = collect(range(2.0*pi/n, 2.0*pi, length=n) .+ rotation) # ao .- 2.0*pi/n - # inner angles + # inner angles ai = ao .+ (ao[2] - ao[1])/2.0 # enforce angles to be between 0 and 2pi for i = 1:n if ao[i] > 2.0*pi ao[i] -= 2.0*pi - end + end if ai[i] > 2.0*pi ai[i] -= 2.0*pi end end - # outer x values + # outer x values xo = ro.*cos.(ao) - # outer y values + # outer y values yo = ro.*sin.(ao) - # inner x values + # inner x values xi = ri.*cos.(ai) # inner y values @@ -1658,10 +1658,8 @@ function star_boundary(n, ri, ro, rotation=0.0) vertices[k,:] = point k += 1 end - end + end # return return round.(vertices, digits=6) end - - diff --git a/src/wake_combination_models.jl b/src/wake_combination_models.jl index ef811d0..fe03329 100644 --- a/src/wake_combination_models.jl +++ b/src/wake_combination_models.jl @@ -1,27 +1,15 @@ abstract type AbstractWakeCombinationModel end struct LinearFreestreamSuperposition <: AbstractWakeCombinationModel - # Lissaman 1979 - # new_deficit_sum = old_deficit_sum + wind_speed*deltav - end struct SumOfSquaresFreestreamSuperposition <: AbstractWakeCombinationModel - # Katic et al. 1986 - # new_deficit_sum = sqrt(old_deficit_sum**2 + (wind_speed*deltav)**2) - end struct SumOfSquaresLocalVelocitySuperposition <: AbstractWakeCombinationModel - # Voutsinas 1990 - # new_deficit_sum = sqrt(old_deficit_sum**2 + (turb_inflow*deltav)**2) - end struct LinearLocalVelocitySuperposition <: AbstractWakeCombinationModel - # Niayifar and Porte Agel 2015, 2016 - # new_deficit_sum = old_deficit_sum + turb_inflow*deltav - end function check_negative_deficits!(new_deficit_sum, wind_speed) @@ -36,18 +24,18 @@ function wake_combination_model(deltav, wind_speed, turb_inflow, old_deficit_sum new_deficit_sum = old_deficit_sum + wind_speed*deltav check_negative_deficits!(new_deficit_sum, wind_speed) - + return new_deficit_sum end function wake_combination_model(deltav, wind_speed, turb_inflow, old_deficit_sum, model::SumOfSquaresFreestreamSuperposition) # Katic et al. 1986 - + new_deficit_sum = nansafesqrt(old_deficit_sum^2 + (wind_speed*deltav)^2) - + check_negative_deficits!(new_deficit_sum, wind_speed) - + return new_deficit_sum end @@ -67,7 +55,7 @@ function wake_combination_model(deltav, wind_speed, turb_inflow, old_deficit_sum # Niayifar and Porte Agel 2015, 2016 new_deficit_sum = old_deficit_sum + turb_inflow*deltav - + check_negative_deficits!(new_deficit_sum, turb_inflow) return new_deficit_sum diff --git a/src/wake_deficit_models.jl b/src/wake_deficit_models.jl index 9790ec4..60007a3 100644 --- a/src/wake_deficit_models.jl +++ b/src/wake_deficit_models.jl @@ -22,9 +22,9 @@ Container for parameters related to the Jensen Cosine deficit model - `alpha::Float`: parameter controlling the wake deficit decay rate. Default value is 0.1 - `beta::Float`: parameter controlling the width of the cosine function. Default value is 20.0 deg., given in radians. """ -struct JensenCosine{TF,ATF} <: AbstractWakeDeficitModel - alpha::TF - beta::TF +struct JensenCosine{TF1,TF2,ATF} <: AbstractWakeDeficitModel + alpha::TF1 + beta::TF2 wec_factor::ATF end JensenCosine() = JensenCosine(0.1, 20.0*pi/180.0, [1.0]) @@ -43,12 +43,12 @@ Container for parameters related to the MultiZone deficit model - `aU::Float`: parameter impacting the wake deficit decay for a constant wake deflection. Default value is 5.0. - `bU::Float`: parameter changing the wake deficit decay under yawed conditions. Default value is 1.66. """ -struct MultiZone{ATF, TF} <: AbstractWakeDeficitModel - me::ATF - ke::TF - MU::ATF - aU::TF - bU::TF +struct MultiZone{ATF1,ATF2,TF1,TF2,TF3} <: AbstractWakeDeficitModel + me::ATF1 + ke::TF1 + MU::ATF2 + aU::TF2 + bU::TF3 end MultiZone() = MultiZone([-0.5 0.22 1.0], 0.065, [0.5 1.0 5.5], 5.0, 1.66) @@ -77,11 +77,11 @@ Container for parameters related to the Gaussian deficit model with yaw presente - `beta_star::Float`: parameter controlling the impact of the thrust coefficient on the length of the near wake. Default value is 0.154. - `interpolation::Bool`: boolean stating if the the near wake should be interpolated. Default value is true. """ -struct GaussYaw{TF, ATF, BO} <: AbstractWakeDeficitModel - horizontal_spread_rate::TF - vertical_spread_rate::TF - alpha_star::TF - beta_star::TF +struct GaussYaw{TF1, TF2, TF3, TF4, ATF, BO} <: AbstractWakeDeficitModel + horizontal_spread_rate::TF1 + vertical_spread_rate::TF2 + alpha_star::TF3 + beta_star::TF4 wec_factor::ATF interpolate_sigma::BO end @@ -101,11 +101,11 @@ Container for parameters related to the Gaussian deficit model with yaw presente - `beta_star::Float`: parameter controlling the impact of the thrust coefficient on the length of the near wake. Default value is 0.154. - `interpolation::Bool`: boolean stating if the the near wake should be interpolated. Default value is true. """ -struct GaussYawVariableSpread{TF, ATF, BO} <: AbstractWakeDeficitModel - alpha_star::TF - beta_star::TF - k1::TF - k2::TF +struct GaussYawVariableSpread{TF1, TF2, TF3, TF4, ATF, BO} <: AbstractWakeDeficitModel + alpha_star::TF1 + beta_star::TF2 + k1::TF3 + k2::TF4 wec_factor::ATF interpolate_sigma::BO end @@ -146,14 +146,14 @@ Container for parameters related to the Cumulative Curl model used in FLORIS v3 - `c_f::Float`: Default value is 2.41 - 'wec_factor': paramter for the wake expansion continuation Default is [1.0] """ -struct CumulativeCurl{TF,ATF} <: AbstractWakeDeficitModel - a_s::TF - b_s::TF - c_s1::TF - c_s2::TF - a_f::TF - b_f::TF - c_f::TF +struct CumulativeCurl{T1,T2,T3,T4,T5,T6,T7,ATF} <: AbstractWakeDeficitModel + a_s::T1 + b_s::T2 + c_s1::T3 + c_s2::T4 + a_f::T5 + b_f::T6 + c_f::T7 wec_factor::ATF end CumulativeCurl() = CumulativeCurl(0.179367259, 0.0118889215, 0.0563691592, 0.13290157, 3.11, -0.68, 2.41, [1.0]) @@ -165,14 +165,14 @@ Computes the wake deficit according to the original Jensen top hat wake model, f "A Note on Wind Generator Interaction" by N.O. Jensen (1983) # Arguments -- `locx::Float`: x coordinate where wind speed is calculated +- `locx::Float`: x coordinate where wind speed is calculated - `locy::Float`: y coordinate where wind speed is calculated - `locz::Float`: z coordinate where wind speed is calculated - `turbine_x::Array(Float)`: vector containing x coordinates for all turbines in farm - `turbine_y::Array(Float)`: vector containing y coordinates for all turbines in farm - `turbine_z::Array(Float)`: vector containing z coordinates for all turbines in farm -- `deflection_y::Float`: deflection in the y direction of downstream wake -- `deflection_z::Float`: deflection in the z direction of downstream wake +- `deflection_y::Float`: deflection in the y direction of downstream wake +- `deflection_z::Float`: deflection in the z direction of downstream wake - `upstream_turbine_id::Int`: index of the upstream wind turbine creating the wake - `downstream_turbine_id::Int`: index of the downstream turbine feeling the wake (if not referencing a turbine set to zero) - `hub_height::Array(Float)`: vector containing hub heights for all turbines in farm @@ -186,9 +186,9 @@ Computes the wake deficit according to the original Jensen top hat wake model, f """ function wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, deflection_y, deflection_z, upstream_turbine_id, downstream_turbine_id, hub_height, rotor_diameter, turbine_ai, turbine_local_ti, turbine_ct, turbine_yaw, model::JensenTopHat) - + r0 = rotor_diameter[upstream_turbine_id]/2.0 #turbine rotor radius - + if downstream_turbine_id == 0 @@ -207,7 +207,7 @@ function wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, d else loss = 2.0*turbine_ai[upstream_turbine_id]*(r0/r)^2 #equation (2) from the paper totloss = loss - end + end else r1 = rotor_diameter[downstream_turbine_id]/2.0 # find delta x, y, and z. dx is the downstream distance from the upstream turbine hub to @@ -247,14 +247,14 @@ Computes the wake deficit according to the original Jensen cosine wake model, fr "A Note on Wind Generator Interaction" by N.O. Jensen (1983) # Arguments -- `locx::Float`: x coordinate where wind speed is calculated +- `locx::Float`: x coordinate where wind speed is calculated - `locy::Float`: y coordinate where wind speed is calculated - `locz::Float`: z coordinate where wind speed is calculated - `turbine_x::Array(Float)`: vector containing x coordinates for all turbines in farm - `turbine_y::Array(Float)`: vector containing y coordinates for all turbines in farm - `turbine_z::Array(Float)`: vector containing z coordinates for all turbines in farm -- `deflection_y::Float`: deflection in the y direction of downstream wake -- `deflection_z::Float`: deflection in the z direction of downstream wake +- `deflection_y::Float`: deflection in the y direction of downstream wake +- `deflection_z::Float`: deflection in the z direction of downstream wake - `upstream_turbine_id::Int`: index of the upstream wind turbine creating the wake - `downstream_turbine_id::Int`: index of the downstream turbine feeling the wake (if not referencing a turbine set to zero) - `hub_height::Array(Float)`: vector containing hub heights for all turbines in farm @@ -307,14 +307,14 @@ Computes the wake deficit at a given location using the original MultiZone "FLOR "Wind plant power optimization through yaw control using a parametric model for wake effects—a CFD simulation study" by Gebraad et al. (2014) # Arguments -- `locx::Float`: x coordinate where wind speed is calculated +- `locx::Float`: x coordinate where wind speed is calculated - `locy::Float`: y coordinate where wind speed is calculated - `locz::Float`: z coordinate where wind speed is calculated - `turbine_x::Array(Float)`: vector containing x coordinates for all turbines in farm - `turbine_y::Array(Float)`: vector containing y coordinates for all turbines in farm - `turbine_z::Array(Float)`: vector containing z coordinates for all turbines in farm -- `deflection_y::Float`: deflection in the y direction of downstream wake -- `deflection_z::Float`: deflection in the z direction of downstream wake +- `deflection_y::Float`: deflection in the y direction of downstream wake +- `deflection_z::Float`: deflection in the z direction of downstream wake - `upstream_turbine_id::Int`: index of the upstream wind turbine creating the wake - `downstream_turbine_id::Int`: index of the downstream turbine feeling the wake (if not referencing a turbine set to zero) - `hub_height::Array(Float)`: vector containing hub heights for all turbines in farm @@ -330,7 +330,7 @@ function wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, d dt = rotor_diameter[upstream_turbine_id] - + # extract model parameters ke = model.ke @@ -404,7 +404,7 @@ function wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, d c1,c2,c3 = 0,0,0 else del = norm([dxt, dzt]) #distance from wake center to the point of interest - + wake_center_y = (turbine_y[upstream_turbine_id]+deflection_y) wake_center_z = (turbine_z[upstream_turbine_id]+hub_height[upstream_turbine_id]+deflection_z) @@ -420,7 +420,7 @@ function wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, d area = pi*(rotor_diameter[downstream_turbine_id]^2)/4 ovlp[i] = overlap/area end - + Rw = Dw./2 # radius of the wake zones # equations (15, 16, and 17) from the paper calculated for all 3 zones @@ -457,14 +457,14 @@ end Computes the wake deficit at a given location using the Gaussian wake model presented by Bastankhah and Porte-Agel in the paper: "A new analytical model for wind-turbine wakes" (2014) # Arguments -- `locx::Float`: x coordinate where wind speed is calculated +- `locx::Float`: x coordinate where wind speed is calculated - `locy::Float`: y coordinate where wind speed is calculated - `locz::Float`: z coordinate where wind speed is calculated - `turbine_x::Array(Float)`: vector containing x coordinates for all turbines in farm - `turbine_y::Array(Float)`: vector containing y coordinates for all turbines in farm - `turbine_z::Array(Float)`: vector containing z coordinates for all turbines in farm -- `deflection_y::Float`: deflection in the y direction of downstream wake -- `deflection_z::Float`: deflection in the z direction of downstream wake +- `deflection_y::Float`: deflection in the y direction of downstream wake +- `deflection_z::Float`: deflection in the z direction of downstream wake - `upstream_turbine_id::Int`: index of the upstream wind turbine creating the wake - `downstream_turbine_id::Int`: index of the downstream turbine feeling the wake (if not referencing a turbine set to zero) - `hub_height::Array(Float)`: vector containing hub heights for all turbines in farm @@ -537,21 +537,21 @@ end _gauss_yaw_spread_interpolated(dt, k, dx, x0, yaw) Helper function for wake_deficit_model when using the GaussYaw model. Computes the standard deviation of the wake. -with an interpolation on the near wake. +with an interpolation on the near wake. """ function _gauss_yaw_spread_interpolated(dt, k, dx, x0, yaw, xd; interpolate=true) # calculate wake spread if interpolate - if dx > x0 # far wake + if dx > x0 # far wake sigma = _gauss_yaw_spread(dt, k, dx, x0, yaw) else # linear interpolation in the near wakes dx_interpolate = xd+((x0-xd)/(x0))*(dx) sigma = _gauss_yaw_spread(dt, k, dx_interpolate, x0, yaw) end else - if dx > xd # far wake + if dx > xd # far wake sigma = _gauss_yaw_spread(dt, k, dx, x0, yaw) else # use sigma at xd for undefined values sigma = _gauss_yaw_spread(dt, k, xd, x0, yaw) @@ -581,12 +581,12 @@ function _gauss_yaw_model_deficit(dx, dy, dz, dt, yaw, ct, ti, as, bs, ky, kz, w # calculate the length of the potential core (paper eq: 7.3) x0 = _gauss_yaw_potential_core(dt, yaw, ct, as, ti, bs) - # calculate the discontinuity point of the gauss yaw model + # calculate the discontinuity point of the gauss yaw model xd = _gauss_yaw_discontinuity(dt, x0, ky, kz, yaw, ct) - + # calculate horizontal wake spread (paper eq: 7.2) sigma_y = _gauss_yaw_spread_interpolated(dt, ky, dx, x0, yaw, xd) - + # calculate vertical wake spread (paper eq: 7.2) sigma_z = _gauss_yaw_spread_interpolated(dt, kz, dx, x0, 0.0, xd) @@ -615,14 +615,14 @@ end Computes the wake deficit at a given location using the The Gaussian wake model presented by Bastankhah and Porte-Agel in the paper: "Experimental and theoretical study of wind turbine wakes in yawed conditions" (2016) # Arguments -- `locx::Float`: x coordinate where wind speed is calculated +- `locx::Float`: x coordinate where wind speed is calculated - `locy::Float`: y coordinate where wind speed is calculated - `locz::Float`: z coordinate where wind speed is calculated - `turbine_x::Array(Float)`: vector containing x coordinates for all turbines in farm - `turbine_y::Array(Float)`: vector containing y coordinates for all turbines in farm - `turbine_z::Array(Float)`: vector containing z coordinates for all turbines in farm -- `deflection_y::Float`: deflection in the y direction of downstream wake -- `deflection_z::Float`: deflection in the z direction of downstream wake +- `deflection_y::Float`: deflection in the y direction of downstream wake +- `deflection_z::Float`: deflection in the z direction of downstream wake - `upstream_turbine_id::Int`: index of the upstream wind turbine creating the wake - `downstream_turbine_id::Int`: index of the downstream turbine feeling the wake (if not referencing a turbine set to zero) - `hub_height::Array(Float)`: vector containing hub heights for all turbines in farm @@ -667,14 +667,14 @@ Computes the wake deficit at a given location using the The Gaussian wake model The spread rate is adjusted based on local turbulence intensity as in Niayifar and Porte-Agel 2016 # Arguments -- `locx::Float`: x coordinate where wind speed is calculated +- `locx::Float`: x coordinate where wind speed is calculated - `locy::Float`: y coordinate where wind speed is calculated - `locz::Float`: z coordinate where wind speed is calculated - `turbine_x::Array(Float)`: vector containing x coordinates for all turbines in farm - `turbine_y::Array(Float)`: vector containing y coordinates for all turbines in farm - `turbine_z::Array(Float)`: vector containing z coordinates for all turbines in farm -- `deflection_y::Float`: deflection in the y direction of downstream wake -- `deflection_z::Float`: deflection in the z direction of downstream wake +- `deflection_y::Float`: deflection in the y direction of downstream wake +- `deflection_z::Float`: deflection in the z direction of downstream wake - `upstream_turbine_id::Int`: index of the upstream wind turbine creating the wake - `downstream_turbine_id::Int`: index of the downstream turbine feeling the wake (if not referencing a turbine set to zero) - `hub_height::Array(Float)`: vector containing hub heights for all turbines in farm @@ -718,14 +718,14 @@ Computes the wake deficit at a given location using the Gaussian wake model pres as modified for IEA Task 37 Case Studies 3 and 4 # Arguments -- `locx::Float`: x coordinate where wind speed is calculated +- `locx::Float`: x coordinate where wind speed is calculated - `locy::Float`: y coordinate where wind speed is calculated - `locz::Float`: z coordinate where wind speed is calculated - `turbine_x::Array(Float)`: vector containing x coordinates for all turbines in farm - `turbine_y::Array(Float)`: vector containing y coordinates for all turbines in farm - `turbine_z::Array(Float)`: vector containing z coordinates for all turbines in farm -- `deflection_y::Float`: deflection in the y direction of downstream wake -- `deflection_z::Float`: deflection in the z direction of downstream wake +- `deflection_y::Float`: deflection in the y direction of downstream wake +- `deflection_z::Float`: deflection in the z direction of downstream wake - `upstream_turbine_id::Int`: index of the upstream wind turbine creating the wake - `downstream_turbine_id::Int`: index of the downstream turbine feeling the wake (if not referencing a turbine set to zero) - `hub_height::Array(Float)`: vector containing hub heights for all turbines in farm @@ -738,7 +738,7 @@ Computes the wake deficit at a given location using the Gaussian wake model pres """ function wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, deflection_y, deflection_z, upstream_turbine_id, downstream_turbine_id, hub_height, rotor_diameter, turbine_ai, turbine_local_ti, turbine_ct, turbine_yaw, model::GaussSimple) - + dx = locx-turbine_x[upstream_turbine_id] dy = locy-(turbine_y[upstream_turbine_id]+deflection_y) @@ -750,7 +750,7 @@ function wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, d k = model.k wf = model.wec_factor[1] - # calculate loss + # calculate loss sigmay = k*dx + dt/sqrt(8.0) radical = 1.0 - ct/(8.0*(sigmay^2)/(dt^2)) exponent = -0.5*(dy/(wf*sigmay))^2 @@ -767,14 +767,14 @@ end # "Addressing deep array effects and impacts to wake steering with the cumulative-curl wake model" by Bay, Christopher (2022) (https://doi.org/10.5194/wes-2022-17) # # Arguments -# - `locx::Float`: x coordinate where wind speed is calculated +# - `locx::Float`: x coordinate where wind speed is calculated # - `locy::Float`: y coordinate where wind speed is calculated # - `locz::Float`: z coordinate where wind speed is calculated # - `turbine_x::Array(Float)`: vector containing x coordinates for all turbines in farm # - `turbine_y::Array(Float)`: vector containing y coordinates for all turbines in farm # - `turbine_z::Array(Float)`: vector containing z coordinates for all turbines in farm -# - `deflection_y::Float`: deflection in the y direction of downstream wake -# - `deflection_z::Float`: deflection in the z direction of downstream wake +# - `deflection_y::Float`: deflection in the y direction of downstream wake +# - `deflection_z::Float`: deflection in the z direction of downstream wake # - `upstream_turbine_id::Int`: index of the upstream wind turbine creating the wake # - `downstream_turbine_id::Int`: index of the downstream turbine feeling the wake (if not referencing a turbine set to zero) # - `hub_height::Array(Float)`: vector containing hub heights for all turbines in farm @@ -788,4 +788,4 @@ end # """ # function wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, deflection_y, deflection_z, upstream_turbine_id, downstream_turbine_id, hub_height, rotor_diameter, turbine_ai, turbine_local_ti, turbine_ct, turbine_yaw, model::CumulativeCurl) # # extract model properties -# end \ No newline at end of file +# end diff --git a/src/wake_deflection_models.jl b/src/wake_deflection_models.jl index 4135eab..56630be 100644 --- a/src/wake_deflection_models.jl +++ b/src/wake_deflection_models.jl @@ -20,17 +20,16 @@ Container for parameters related to the Gaussian deflection model presented by B - `beta_star::Float`: parameter controlling the impact of the thrust coefficient on the length of the near wake. Default value is 0.154. - `interpolation::Bool`: boolean stating if the the near wake should be interpolated. Default value is true. """ -struct GaussYawDeflection{TF, BO} <: AbstractWakeDeflectionModel - horizontal_spread_rate::TF - vertical_spread_rate::TF - alpha_star::TF - beta_star::TF +struct GaussYawDeflection{TF1, TF2, TF3, TF4, BO} <: AbstractWakeDeflectionModel + horizontal_spread_rate::TF1 + vertical_spread_rate::TF2 + alpha_star::TF3 + beta_star::TF4 interpolate_sigma::BO end GaussYawDeflection() = GaussYawDeflection(0.022, 0.022, 2.32, 0.154, true) GaussYawDeflection(interp) = GaussYawDeflection(0.022, 0.022, 2.32, 0.154, interp) GaussYawDeflection(a,b,c,d) = GaussYawDeflection(a, b, c, d, true) -GaussYawDeflection(a,b,c,d,interp) = GaussYawDeflection(a, b, c, d, interp) """ GaussYawDeflectionVariableSpread(alpha_star, beta_star, k1, k2, interpolation) @@ -44,11 +43,11 @@ Container for parameters related to the Gaussian deflection model with yaw prese - `k2::Float`: second parameter tuning wake spread as based on turbulence intensity - `interpolation::Bool`: boolean stating if the the near wake should be interpolated. Default value is true. """ -struct GaussYawVariableSpreadDeflection{TF, BO} <: AbstractWakeDeflectionModel - alpha_star::TF - beta_star::TF - k1::TF - k2::TF +struct GaussYawVariableSpreadDeflection{TF1, TF2, TF3, TF4, BO} <: AbstractWakeDeflectionModel + alpha_star::TF1 + beta_star::TF2 + k1::TF3 + k2::TF4 interpolate_sigma::BO end GaussYawVariableSpreadDeflection() = GaussYawVariableSpreadDeflection(2.32, 0.154, 0.3837, 0.003678, true) @@ -79,10 +78,10 @@ Container for parameters related to the Jiminez deflection model - `ad::Float`:Helps define the horizontal deflection of the wake at 0 deg yaw - `bd::Float`:Helps define the horizontal deflection of the wake due to downwind distance at 0 deg yaw """ -struct MultizoneDeflection{TF} <: AbstractWakeDeflectionModel - horizontal_spread_rate::TF - ad::TF - bd::TF +struct MultizoneDeflection{TF1, TF2, TF3} <: AbstractWakeDeflectionModel + horizontal_spread_rate::TF1 + ad::TF2 + bd::TF3 end MultizoneDeflection() = MultizoneDeflection(0.15, -4.5, -0.01) @@ -175,7 +174,7 @@ function wake_deflection_model(locx, locy, locz, turbine_x, turbine_yaw, turbine end function _bpa_theta_0(yaw, ct) - + theta0 = (0.3*yaw/cos(yaw))*(1.0-sqrt(1.0-ct*cos(yaw))) return theta0 @@ -218,16 +217,16 @@ function wake_deflection_model(locx, locy, locz, turbine_x, turbine_yaw, turbine # [1] eqn 7.4 x0 = _gauss_yaw_potential_core(diam, yaw, ct, as, ti, bs) - # calculate the discontinuity point of the gauss yaw model + # calculate the discontinuity point of the gauss yaw model xd = _gauss_yaw_discontinuity(diam, x0, ky, kz, yaw, ct) - + # calculate horizontal wake spread (paper eq: 7.2) sigmay = _gauss_yaw_spread_interpolated(diam, ky, dx, x0, yaw, xd) # calculate vertical wake spread (paper eq: 7.2) sigmaz = _gauss_yaw_spread_interpolated(diam, kz, dx, x0, 0.0, xd) - + y_deflection = _bpa_deflection(diam, ct, yaw, ky, kz, sigmay, sigmaz, theta0, x0) return y_deflection @@ -264,9 +263,9 @@ function wake_deflection_model(locx, locy, locz, turbine_x, turbine_yaw, turbine # [1] eqn 7.4 x0 = _gauss_yaw_potential_core(dt, yaw, ct, as, ti, bs) - # calculate the discontinuity point of the gauss yaw model + # calculate the discontinuity point of the gauss yaw model xd = _gauss_yaw_discontinuity(dt, x0, ky, kz, yaw, ct) - + # calculate horizontal wake spread (paper eq: 7.2) sigma_y = _gauss_yaw_spread_interpolated(dt, ky, dx, x0, yaw, xd) @@ -277,4 +276,4 @@ function wake_deflection_model(locx, locy, locz, turbine_x, turbine_yaw, turbine y_deflection = _bpa_deflection(dt, ct, yaw, ky, kz, sigma_y, sigma_z, theta0, x0) return y_deflection -end \ No newline at end of file +end diff --git a/src/wind_resources.jl b/src/wind_resources.jl index 40e5d34..dea64cb 100644 --- a/src/wind_resources.jl +++ b/src/wind_resources.jl @@ -16,14 +16,14 @@ abstract type AbstractWindResourceModel end - `ambient_ti::Array{Float,1}`: an array of the ambient turbulence intensity for each wind direction - `wind_shear_model::Array{AbstractWindShearModel}(1)`: contains a struct defining the desired turbulence intensity model """ -struct DiscretizedWindResource{AF, TF, ASM} <: AbstractWindResourceModel +struct DiscretizedWindResource{AF1, AF2, AF3, AF4, AF5, TF, ASM} <: AbstractWindResourceModel - wind_directions::AF - wind_speeds::AF - wind_probabilities::AF - measurement_heights::AF + wind_directions::AF1 + wind_speeds::AF2 + wind_probabilities::AF3 + measurement_heights::AF4 air_density::TF - ambient_tis::AF + ambient_tis::AF5 wind_shear_model::ASM end @@ -48,7 +48,7 @@ function rediscretize_windrose(windrosein::DiscretizedWindResource, ndirectionbi probabilityspline = Akima(splinedirs, [windrosein.wind_probabilities; windrosein.wind_probabilities; windrosein.wind_probabilities]) heightspline = Akima(splinedirs, [windrosein.measurement_heights; windrosein.measurement_heights; windrosein.measurement_heights]) ambienttispline = Akima(splinedirs, [windrosein.ambient_tis; windrosein.ambient_tis; windrosein.ambient_tis]) - + # get new interpolated wind rose attributes directionsnew = collect(range(start,2*pi+start-2*pi/ndirectionbins,length=ndirectionbins)) if averagespeed @@ -60,10 +60,10 @@ function rediscretize_windrose(windrosein::DiscretizedWindResource, ndirectionbi heightsnew = heightspline.(directionsnew) ambient_tis_new = ambienttispline.(directionsnew) - # re-normalize probabilites + # re-normalize probabilites probabilitiesnew = probabilitiesnew./sum(probabilitiesnew) - # create new wind rose + # create new wind rose windroseout = DiscretizedWindResource(directionsnew, speedsnew, probabilitiesnew, heightsnew, windrosein.air_density, ambient_tis_new, windrosein.wind_shear_model) return windroseout diff --git a/src/wind_shear_models.jl b/src/wind_shear_models.jl index 8e58997..839f65e 100644 --- a/src/wind_shear_models.jl +++ b/src/wind_shear_models.jl @@ -11,11 +11,11 @@ Ground height may be tuned because the power law does not always hold near the g - `ground_height::Float`: height of the ground (typically zero) - `shear_order::Bool`: when shear should be calculated. Can be "first", "last", or "none" """ -struct PowerLawWindShear{TF, TS} <: AbstractWindShearModel - +struct PowerLawWindShear{TF1, TF2, TS} <: AbstractWindShearModel + # model parameter - shear_exponent::TF - ground_height::TF + shear_exponent::TF1 + ground_height::TF2 shear_order::TS end @@ -32,7 +32,7 @@ not always hold near the ground. # Arguments - `locz::Float`: height of desired velocity - `reference_velocity::Float`: known velocity at reference_height -- `reference_height::Float`: height of known velocity +- `reference_height::Float`: height of known velocity - `ground_height::Float`: height of the ground (typically zero) - `model::AbstractWindShearModel`: wind shear model to use for calculations """ @@ -41,7 +41,7 @@ function adjust_for_wind_shear(locz, reference_velocity, reference_height, groun # initialize adjusted wind speed to zero adjusted_wind_speed = 0.0 shear_exp = model.shear_exponent - + # check that the point of interest is above ground level if locz >= ground_height # adjusted wind speed for wind shear if point is above ground @@ -60,10 +60,10 @@ end # if loc[3] >= ground_height # # adjusted wind speed for wind shear if point is above ground # adjusted_wind_speed = point_velocity_no_shear*((loc[3]-ground_height)/(reference_height-ground_height))^shear_exp -# else +# else # # if the point of interest is below ground, set the wind speed to 0.0 # adjusted_wind_speed = 0.0 # end # return adjusted_wind_speed -# end \ No newline at end of file +# end diff --git a/src/windfarms.jl b/src/windfarms.jl index ee05769..0d2f158 100644 --- a/src/windfarms.jl +++ b/src/windfarms.jl @@ -73,6 +73,3 @@ end # boundary = PolygonBoundary(vertices, normals) # return boundary # end - - - diff --git a/test/runtests.jl b/test/runtests.jl index 868f332..61a9564 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -5,12 +5,32 @@ using Distributed using DelimitedFiles using LinearAlgebra using FLOWMath: linear -using Distributed using YAML using ForwardDiff using FiniteDiff @testset ExtendedTestSet "all tests" begin + # @testset "type stability" begin + # @testset "AEP Calculation" begin + # include("model_sets/model_set_6.jl") + # function testAEP() + # return ff.calculate_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, + # hub_height, turbine_yaw, ct_models, generator_efficiency, cut_in_speed, + # cut_out_speed, rated_speed, rated_power, windresource, power_models, model_set)/1e9 + # end + # function checkstability() + # try + # @inferred testAEP() + # return true + # catch err + # println(err) + # return false + # end + # end + # @test checkstability() + # end + # end + @testset "cost_models" begin Parameters = ff.Levelized() rotor_diameter = [70] @@ -75,7 +95,7 @@ using FiniteDiff @testset "latitude longitude to xy" begin latitude = [59.3293, 59.8586] - longitude = [18.0686, 17.6389] + longitude = [18.0686, 17.6389] utm_zone = 33 x, y = ff.latlong_to_xy(latitude, longitude, utm_zone) @@ -185,23 +205,23 @@ using FiniteDiff boundary_file_name = string("./inputfiles/iea37-boundary-cs3.yaml") boundary_vertices2 = ff.get_boundary_yaml(boundary_file_name) boundary_normals2 = ff.boundary_normals_calculator(boundary_vertices2) - correct_normals2 = [0.9829601758936983 -0.1838186405319916; - 0.9934614633172962 -0.11416795042154541; - 0.9987121579438882 -0.050734855622757584; - 0.9998686751666075 -0.01620593781838486; - 0.9999954987444023 0.0030004151269687495; - -0.9998078216567232 -0.019604074934516894; - -0.6957179389375846 -0.718315076718037; - -0.6957275377423737 -0.7183057797532565; - -0.8019887481131871 0.5973391397688945; - 0.5138086803485797 0.8579047965820281; - 0.4252760929807897 0.905063668886888; - 0.2645057513093967 0.9643841078762402; - -0.0684295708121141 0.9976559496331737; - -0.39636379138742883 0.9180935381958544; - -0.6828023205475376 0.7306031693435896; - -0.7996740386176392 0.6004343694034798; - -0.8578802011411015 0.5138497450520954; + correct_normals2 = [0.9829601758936983 -0.1838186405319916; + 0.9934614633172962 -0.11416795042154541; + 0.9987121579438882 -0.050734855622757584; + 0.9998686751666075 -0.01620593781838486; + 0.9999954987444023 0.0030004151269687495; + -0.9998078216567232 -0.019604074934516894; + -0.6957179389375846 -0.718315076718037; + -0.6957275377423737 -0.7183057797532565; + -0.8019887481131871 0.5973391397688945; + 0.5138086803485797 0.8579047965820281; + 0.4252760929807897 0.905063668886888; + 0.2645057513093967 0.9643841078762402; + -0.0684295708121141 0.9976559496331737; + -0.39636379138742883 0.9180935381958544; + -0.6828023205475376 0.7306031693435896; + -0.7996740386176392 0.6004343694034798; + -0.8578802011411015 0.5138497450520954; 0.42552559023380465 0.9049463918134445] @test boundary_normals2 ≈ correct_normals2 atol=1E-6 @@ -220,10 +240,10 @@ using FiniteDiff ytest = [0.15496810158044924, -0.3958382330389885, 0.40710859551189754, -0.10572443397953969, -0.36940158024655595, 0.7347989934111504, -0.7340708874922061, 0.30479782203943484, 0.36091623014218815, -0.9057342725556136] @test x ≈ xtest atol=1E-6 @test y ≈ ytest atol=1E-6 - + end - @testset "grid_points" begin + @testset "grid_points" begin # test with grid at size of rotor-swept area including perimeter y, z = ff.grid_points(9) @@ -270,10 +290,10 @@ using FiniteDiff ytest = [0, -1, 0, 1, 0] @test x ≈ xtest atol=1E-6 @test y ≈ ytest atol=1E-6 - + end - @testset "wake_count_iec" begin + @testset "wake_count_iec" begin turbinex = [0, 100, 200, 300] turbiney = zeros(4) diameter = ones(4).*40 @@ -284,7 +304,7 @@ using FiniteDiff wake_count = ff.wake_count_iec(turbinex, turbiney, winddirection, diameter) @test wake_count == correct_wake_count - # test with turbines all in free-stream + # test with turbines all in free-stream winddirection = 0.0 correct_wake_count = zeros(4) wake_count = ff.wake_count_iec(turbinex, turbiney, winddirection, diameter) @@ -297,7 +317,7 @@ using FiniteDiff @test wake_count == correct_wake_count end - @testset "find_upstream_turbines" begin + @testset "find_upstream_turbines" begin turbinex = [0.0 100.0] turbiney = [0.0 0.0] diameter = [20.0 20.0] @@ -351,11 +371,11 @@ using FiniteDiff c = ff.pointinpolygon([0.0, 0.0], vertices, return_distance=false) @test c == -1 - # check that ArgumentError occurs if point is given in ints and distance is desired + # check that ArgumentError occurs if point is given in ints and distance is desired @test_throws ArgumentError ff.pointinpolygon([-5, 5], vertices, return_distance=true) # almost a triangle but four sided shape - vertices = [0.0 0.0; 0.0 10.0; 4.0 4.0; 10.0 2.0] # almost a triangle + vertices = [0.0 0.0; 0.0 10.0; 4.0 4.0; 10.0 2.0] # almost a triangle # test point directly below acute angle and outside of shape point = [10.0, 0.0] @@ -375,10 +395,10 @@ using FiniteDiff @testset "point in polygon derivatives" begin - # set up turbine location for testing + # set up turbine location for testing point = [0.0, 0.0] - # set up simple square boundary for testing + # set up simple square boundary for testing boundaryvertices = reverse([-1.0 -1.0; -1.0 1.0; 1.0 1.0; 1.0 -1.0]) boundarynormals = ff.boundary_normals_calculator(boundaryvertices) @@ -390,7 +410,7 @@ using FiniteDiff derivad = ForwardDiff.gradient(pointinpolygon_diff, point) @test derivad ≈ derivfd atol = 1E-6 - # test correct derivative inside to face + # test correct derivative inside to face point[1] = 0.1 derivfd = FiniteDiff.finite_difference_gradient(pointinpolygon_diff, point, Val{:central}) derivad = ForwardDiff.gradient(pointinpolygon_diff, point) @@ -470,8 +490,8 @@ using FiniteDiff end - @testset "nansafesqrt" begin - + @testset "nansafesqrt" begin + a1 = 9.0 a2 = 0.0 a3 = eps()*1E-10 @@ -486,8 +506,6 @@ using FiniteDiff @test isapprox(ff.nansafesqrt(a3), a3*sqrt(eps())/eps()) end - - end @testset "Optimization functions" begin @@ -555,7 +573,7 @@ using FiniteDiff testing_x = [cc_l, 100.0, cc_r, cc_l, 100.0, cc_r, cc_l, 100.0, cc_r] testing_y = [cc_r, cc_r, cc_r, 100.0, 100.0, 100.0, cc_l, cc_l, cc_l] - + test_values = [ -cc_r, -cc_l, -0.0, -cc_d, -100.0, -100.0, -cc_l, -cc_r, -cc_l, -cc_r, -0.0, -cc_d, @@ -612,7 +630,7 @@ using FiniteDiff # @test ff.splined_boundary(testing_x, testing_y, bndry_x_clsd, bndry_y_clsd, bndry_corner_indcies) == test_values #atol=1E-3 #-- Multi-turbine concave boundary test --# - + end @testset "Variable Reduction (Boundary)" begin @@ -650,7 +668,7 @@ using FiniteDiff @test testing_x[1] == test_values[1] @test testing_y[1] == test_values[2] @test num_leftover == test_values[3] - end + end @testset "One Turbine, Circular Boundary, With Start Distance" begin #-- One-turbine circular boundary, zero start distance --# @@ -795,41 +813,41 @@ using FiniteDiff @testset "ray casting boundary distances - single region" begin - # set up turbine location for testing + # set up turbine location for testing turbinex = [0.0] turbiney = [0.0] - # set up simple square boundary for testing + # set up simple square boundary for testing boundaryvertices = [-1.0 -1.0; -1.0 1.0; 1.0 1.0; 1.0 -1.0] boundarynormals = ff.boundary_normals_calculator(boundaryvertices) - # test correct sign (negative) for inside + # test correct sign (negative) for inside boundarydistance = ff.ray_casting_boundary(boundaryvertices, boundarynormals, turbinex, turbiney) @test sign(boundarydistance[1]) == -1 - # test correct sign (positive) for outside + # test correct sign (positive) for outside turbinex = [-2.0] boundarydistance = ff.ray_casting_boundary(boundaryvertices, boundarynormals, turbinex, turbiney) @test sign(boundarydistance[1]) == 1 - # test correct distance inside + # test correct distance inside turbinex = [0.0] boundarydistance = ff.ray_casting_boundary(boundaryvertices, boundarynormals, turbinex, turbiney) @test boundarydistance[1] ≈ -1 atol = 1E-2 - # test correct distance outside + # test correct distance outside turbinex = [2.0] turbiney = [0.0] boundarydistance = ff.ray_casting_boundary(boundaryvertices, boundarynormals, turbinex, turbiney) @test boundarydistance[1] ≈ 1 atol = 1E-2 - # test correct distance on vertex + # test correct distance on vertex turbinex = [1.0] turbiney = [1.0] boundarydistance = ff.ray_casting_boundary(boundaryvertices, boundarynormals, turbinex, turbiney) @test boundarydistance[1] ≈ 0 atol = 1E-3 - # test correct distance on face + # test correct distance on face turbinex = [1.0] turbiney = [0.0] boundarydistance = ff.ray_casting_boundary(boundaryvertices, boundarynormals, turbinex, turbiney) @@ -839,11 +857,11 @@ using FiniteDiff @testset "ray casting boundary distances - multi region pre-defined" begin - # set up turbine location for testing + # set up turbine location for testing turbinex = [0.0] turbiney = [0.0] - # set up simple square boundary for testing + # set up simple square boundary for testing boundaryvertices1 = [-1.0 -1.0; -1.0 1.0; 1.0 1.0; 1.0 -1.0] boundaryvertices2 = [3.0 -1.0; 3.0 1.0; 5.0 1.0; 5.0 -1.0] boundaryvertices = [boundaryvertices1, boundaryvertices2] @@ -929,11 +947,11 @@ using FiniteDiff @testset "ray casting boundary distances - multi region not pre-defined" begin - # set up turbine location for testing + # set up turbine location for testing turbinex = [0.0] turbiney = [0.0] - # set up simple square boundary for testing + # set up simple square boundary for testing boundaryvertices1 = [-1.0 -1.0; -1.0 1.0; 1.0 1.0; 1.0 -1.0] boundaryvertices2 = [3.0 -1.0; 3.0 1.0; 5.0 1.0; 5.0 -1.0] boundaryvertices = [boundaryvertices1, boundaryvertices2] @@ -1007,11 +1025,11 @@ using FiniteDiff @testset "ray casting boundary derivatives - single region" begin - # set up turbine location for testing + # set up turbine location for testing turbinex = [0.0] turbiney = [0.0] - # set up simple square boundary for testing + # set up simple square boundary for testing boundaryvertices = [-1.0 -1.0; -1.0 1.0; 1.0 1.0; 1.0 -1.0] boundarynormals = ff.boundary_normals_calculator(boundaryvertices) @@ -1071,7 +1089,7 @@ using FiniteDiff # derivad = ForwardDiff.jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]]) # @test derivad ≈ derivfd rtol = 1E-4 - # test correct derivative on face + # test correct derivative on face turbinex = [1.0] turbiney = [0.0] derivfd = FiniteDiff.finite_difference_jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]], Val{:central}, absstep=1E-1) @@ -1082,16 +1100,16 @@ using FiniteDiff @testset "ray casting boundary derivatives - multiple regions pre-defined region" begin - # set up turbine location for testing + # set up turbine location for testing turbinex = [0.0] turbiney = [0.0] - # set up simple square boundary for testing + # set up simple square boundary for testing boundaryvertices1 = [-1.0 -1.0; -1.0 1.0; 1.0 1.0; 1.0 -1.0] boundaryvertices2 = [3.0 -1.0; 3.0 1.0; 5.0 1.0; 5.0 -1.0] boundaryvertices = [boundaryvertices1, boundaryvertices2] boundarynormals = ff.boundary_normals_calculator(boundaryvertices, nboundaries=2) - + # set up function for getting AD derivatives ray_casting_boundary_diff(x) = ff.ray_casting_boundary(boundaryvertices, boundarynormals, x[1], x[2], discrete=true, regions=ff.ray_casting_boundary(boundaryvertices, boundarynormals, x[1], x[2], discrete=true, return_region=true)[2]) @@ -1159,14 +1177,14 @@ using FiniteDiff # derivad = ForwardDiff.jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]]) # @test derivad ≈ derivfd rtol = 1E-4 - # test correct derivative on edge region 1 + # test correct derivative on edge region 1 turbinex = [-1.0] turbiney = [0.0] derivfd = FiniteDiff.finite_difference_jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]], Val{:central}, relstep=1E-1) derivad = ForwardDiff.jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]]) @test derivad ≈ derivfd atol = 1E-8 - # test correct derivative on edge region 2 + # test correct derivative on edge region 2 turbinex = [3.0] turbiney = [0.0] derivfd = FiniteDiff.finite_difference_jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]], Val{:central}, relstep=1E-1) @@ -1177,11 +1195,11 @@ using FiniteDiff @testset "ray casting boundary derivatives - multiple regions region not pre-defined" begin - # set up turbine location for testing + # set up turbine location for testing turbinex = [0.0] turbiney = [0.0] - # set up simple square boundary for testing + # set up simple square boundary for testing boundaryvertices1 = [-1.0 -1.0; -1.0 1.0; 1.0 1.0; 1.0 -1.0] boundaryvertices2 = [3.0 -1.0; 3.0 1.0; 5.0 1.0; 5.0 -1.0] boundaryvertices = [boundaryvertices1, boundaryvertices2] @@ -1254,14 +1272,14 @@ using FiniteDiff # derivad = ForwardDiff.jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]]) # @test derivad ≈ derivfd rtol = 1E-4 - # test correct derivative on edge region 1 + # test correct derivative on edge region 1 turbinex = [-1.0] turbiney = [0.0] derivfd = FiniteDiff.finite_difference_jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]], Val{:central}, relstep=1E-1) derivad = ForwardDiff.jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]]) @test derivad ≈ derivfd atol = 1E-8 - # test correct derivative on edge region 2 + # test correct derivative on edge region 2 turbinex = [3.0] turbiney = [0.0] derivfd = FiniteDiff.finite_difference_jacobian(ray_casting_boundary_diff, [turbinex[1] turbiney[1]], Val{:central}, relstep=1E-1) @@ -1340,7 +1358,7 @@ using FiniteDiff hub_height, turbine_yaw, ct_models, generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, windresource, power_models, model_set, rotor_sample_points_y=rotor_points_y,rotor_sample_points_z=rotor_points_z, hours_per_year=365.0*24.0) - + @test aep/1E6 ≈ 938573.62950 rtol=1E-6 end @@ -1355,7 +1373,7 @@ using FiniteDiff hub_height, turbine_yaw, ct_models, generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, windresource, power_models, model_set, rotor_sample_points_y=rotor_points_y,rotor_sample_points_z=rotor_points_z, hours_per_year=365.0*24.0) - + @test aep/1E6 ≈ 2.851096412519999780E6 rtol=1E-6 end @@ -1365,7 +1383,7 @@ using FiniteDiff # import model set with wind farm and related details include("./model_sets/model_set_7_ieacs3.jl") - + state_aeps = ff.calculate_state_aeps(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, ct_models, generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, windresource, power_models, model_set; @@ -1382,19 +1400,19 @@ using FiniteDiff 32035.08418, 52531.37389, 47035.14700, 46848.21422, 45107.13416, 53877.69698, 68105.50430, 69587.76656, 73542.89319, 69615.74101, 66752.31531, 73027.78883, 60187.14103, 59847.98304, 38123.29869] rtol=1E-6 - + end @testset "Test AEP on single turbine" begin # import model set with wind farm and related details include("./model_sets/model_set_7_ieacs4_single_turbine.jl") - + aep = ff.calculate_aep(turbine_x, turbine_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, ct_models, generator_efficiency, cut_in_speed, cut_out_speed, rated_speed, rated_power, windresource, power_models, model_set; rotor_sample_points_y=[0.0], rotor_sample_points_z=[0.0], hours_per_year=365.0*24.0) - + @test aep/1E9 ≈ 42.54982024375254 rtol=1E-6 end @@ -1529,7 +1547,7 @@ using FiniteDiff # test below cut in wt_velocity = 1.0 p = ff.calculate_turbine_power(generator_efficiency[1], cut_in_speed[1], cut_out_speed[1], rated_speed[1], rated_power[1], rotor_diameter[1], wt_velocity, wt_yaw, power_model, air_density) - + @test p ≈ 0.0 atol=1E-6 # region 2 @@ -1559,7 +1577,7 @@ using FiniteDiff # test below cut in wt_velocity = 1.0 p = ff.calculate_turbine_power(generator_efficiency[1], cut_in_speed[1], cut_out_speed[1], rated_speed[1], rated_power[1], rotor_diameter[1], wt_velocity, wt_yaw, power_model, air_density) - + @test p ≈ 0.0 atol=1E-6 # region 2 @@ -1640,14 +1658,14 @@ using FiniteDiff ndirectionbins = 360 startangle = 0.0 wind_resource_new = ff.rediscretize_windrose(wind_resource, ndirectionbins, start=startangle, averagespeed=false) - - # test number of bins + + # test number of bins @test length(wind_resource_new.wind_directions) == ndirectionbins - # test start bin angle + # test start bin angle @test wind_resource_new.wind_directions[1] == startangle - # test probability sum is 1 + # test probability sum is 1 @test sum(wind_resource_new.wind_probabilities) ≈ 1 atol=1E-6 end @@ -1713,18 +1731,18 @@ using FiniteDiff new_deficit_sum = ff.wake_combination_model(deltav, wind_speed, turb_inflow, old_deficit_sum, model) @test new_deficit_sum ≈ result atol=1E-6 - # test correct derivative + # test correct derivative derivfd = FiniteDiff.finite_difference_derivative(deriv_function1, deltav, Val{:central}) derivfad = ForwardDiff.derivative(deriv_function1, deltav) # derivrad = ReverseDiff.gradient(deriv_function, [deltav]) @test derivfad ≈ derivfd atol = 1E-6 # @test derivrad[1] ≈ derivfd atol = 1E-6 - # test correct derivatives at known discontinuity + # test correct derivatives at known discontinuity deltav = 4.823068257348535e-185 wind_speed = 0.9 turb_inflow = 0.9 - old_deficit_sum = 0.0 + old_deficit_sum = 0.0 deriv_function2(x) = ff.wake_combination_model(x, wind_speed, turb_inflow, old_deficit_sum, model) derivfd = FiniteDiff.finite_difference_derivative(deriv_function2, deltav, Val{:central}) @@ -1754,7 +1772,7 @@ using FiniteDiff @test ff.wake_combination_model(deltav, wind_speed, turb_inflow, old_deficit_sum, model) == result end - end + end @testset "Wake Deflection Models" begin @@ -1802,7 +1820,6 @@ using FiniteDiff turbine_inflow_velcity = [8.0] upstream_turbine_id = 1 downstream_turbine_id = 0 - turbine_definition_id = 1 cut_in_speed = 0.0 cut_out_speed = 25.0 rated_speed = 12.0 @@ -1815,7 +1832,6 @@ using FiniteDiff ct_model = ff.ThrustModelConstantCt(ct) power_model = ff.PowerModelConstantCp(constcp) - turbine1 = ff.TurbineDefinition(turbine_definition_id, [rotor_diameter], [hub_height], [cut_in_speed], [rated_speed], [cut_out_speed], [rated_power], [generator_efficiency], [ct_model], [power_model]) model = ff.JiminezYawDeflection(horizontal_spread_rate) dx2p5d = 2.5*rotor_diameter @@ -1860,7 +1876,7 @@ using FiniteDiff turbine_inflow_velcity = [8.0] upstream_turbine_id = 1 downstream_turbine_id = 0 - turbine_definition_id = 1 + cut_in_speed = 0.0 cut_out_speed = 25.0 rated_speed = 12.0 @@ -1870,7 +1886,7 @@ using FiniteDiff ct_model = ff.ThrustModelConstantCt(ct) power_model = ff.PowerModelConstantCp([constcp]) - turbine_definition = ff.TurbineDefinition(turbine_definition_id, [rotor_diameter], [hub_height], [cut_in_speed], [rated_speed], [cut_out_speed], [rated_power], [generator_efficiency], [ct_model], [power_model]) + k_star = 0.022 # [1] p. 525 turbulence_intensity = 0.1 #0.0875 #[1] p. 508 - this value is only specified to be less than 0.1 @@ -1917,7 +1933,7 @@ using FiniteDiff turbine_inflow_velcity = [8.0] upstream_turbine_id = 1 downstream_turbine_id = 0 - turbine_definition_id = 1 + cut_in_speed = 0.0 cut_out_speed = 25.0 rated_speed = 12.0 @@ -1926,7 +1942,7 @@ using FiniteDiff ct_model = ff.ThrustModelConstantCt(ct) power_model = ff.PowerModelConstantCp([constcp]) - turbine_definition = ff.TurbineDefinition(turbine_definition_id, [rotor_diameter], [hub_height], [cut_in_speed], [rated_speed], [cut_out_speed], [rated_power], [generator_efficiency], [ct_model], [power_model]) + turbulence_intensity = 0.07 # this value is just guessed #TODO find data about deflection using this model alpha_star = 2.32 #[1] p. 534 @@ -2007,7 +2023,7 @@ using FiniteDiff @test ff.wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, deflection_y, deflection_z, upstream_turbine_id, downstream_turbine_id, hub_height, rotor_diameter, turbine_ai, turbine_local_ti, turbine_ct, turbine_yaw, wakedeficitmodel) == centerloss100 locy = -(alpha*100. + rotor_diameter[1]/2. + 1E-12) @test ff.wake_deficit_model(locx, locy, locz, turbine_x, turbine_y, turbine_z, deflection_y, deflection_z, upstream_turbine_id, downstream_turbine_id, hub_height, rotor_diameter, turbine_ai, turbine_local_ti, turbine_ct, turbine_yaw, wakedeficitmodel) == 0.0 - + # test overlap function upstream_turbine_id = 2 downstream_turbine_id = 1 @@ -2047,7 +2063,7 @@ using FiniteDiff ambient_ti = 0.1 upstream_turbine_id = 1 downstream_turbine_id = 0 - turbine_definition_id = 1 + alpha = 0.1 @@ -2059,7 +2075,6 @@ using FiniteDiff ct_model = ff.ThrustModelConstantCt(ct) power_model = ff.PowerModelConstantCp([constcp]) - turbine_definition = ff.TurbineDefinition(1, [rotor_diameter], [hub_height], [cut_in_speed], [rated_speed], [cut_out_speed], [rated_power], [generator_efficiency], [ct_model], [power_model]) model = ff.JensenCosine(alpha, beta) centerloss40 = 1. - 4.35/8.1 @@ -2141,8 +2156,8 @@ using FiniteDiff sorted_turbine_index, ct_models, rotor_sample_points_y, rotor_sample_points_z, wind_resource, model_set) - turbine_powers = ff.turbine_powers_one_direction(generator_efficiency, cut_in_speed, cut_out_speed, - rated_speed, rated_power, rotor_diameter, turbine_inflow_velocities, turbine_yaw, air_density, + turbine_powers = ff.turbine_powers_one_direction(generator_efficiency, cut_in_speed, cut_out_speed, + rated_speed, rated_power, rotor_diameter, turbine_inflow_velocities, turbine_yaw, air_density, power_models) @test turbine_powers[1] ≈ 790066 atol=0.1 @@ -2185,7 +2200,7 @@ using FiniteDiff ambient_ti = 0.1 upstream_turbine_id = 1 downstream_turbine_id = 0 - turbine_definition_id = 1 + deflection_y = 0.0 deflection_z = 0.0 @@ -2194,7 +2209,6 @@ using FiniteDiff ct_model = ff.ThrustModelConstantCt(ct) power_model = ff.PowerModelConstantCp([constcp]) - turbine_definition = ff.TurbineDefinition(1, [rotor_diameter], [hub_height], [cut_in_speed], [rated_speed], [cut_out_speed], [rated_power], [generator_efficiency], [ct_model], [power_model]) model = ff.GaussYaw(horizontal_spread_rate, vertical_spread_rate, alpha_star, beta_star) @@ -2288,7 +2302,7 @@ using FiniteDiff # println(ambient_ti) upstream_turbine_id = 1 downstream_turbine_id = 0 - turbine_definition_id = 1 + deflection_y = 0.0 deflection_z = 0.0 @@ -2297,7 +2311,6 @@ using FiniteDiff ct_model = ff.ThrustModelConstantCt(ct) power_model = ff.PowerModelConstantCp([constcp]) - turbine_definition = ff.TurbineDefinition(1, [rotor_diameter], [hub_height], [cut_in_speed], [rated_speed], [cut_out_speed], [rated_power], [generator_efficiency], [ct_model], [power_model]) model = ff.GaussYawVariableSpread(alpha_star, beta_star) @@ -2352,23 +2365,23 @@ using FiniteDiff end - @testset "Cumulative Curl Model" begin + # @testset "Cumulative Curl Model" begin - include("./model_sets/model_set_CumulativeCurl.jl") + # include("./model_sets/model_set_CumulativeCurl.jl") - # These speeds come from the exact same senario run in FLORIS - floris_speeds = [4.47782564 4.22203667 4.87630822 7.24928086 5.3327349 7.24928086 4.87630822 4.10768106 5.89349064 6.30625438 7.79997475 7.99694765 8. 8. 8. 7.99694765 7.79997475 6.30625438 5.89349064 4.02800927 4.13783708 5.85694226 7.80549692 5.84344344 7.98537599 8. 8. 8. 8. 8. 8. 8. 8. 7.98537599 5.84344344 7.80549692 5.85694226 4.13783708] + # # These speeds come from the exact same senario run in FLORIS + # floris_speeds = [4.47782564 4.22203667 4.87630822 7.24928086 5.3327349 7.24928086 4.87630822 4.10768106 5.89349064 6.30625438 7.79997475 7.99694765 8. 8. 8. 7.99694765 7.79997475 6.30625438 5.89349064 4.02800927 4.13783708 5.85694226 7.80549692 5.84344344 7.98537599 8. 8. 8. 8. 8. 8. 8. 8. 7.98537599 5.84344344 7.80549692 5.85694226 4.13783708] - rot_x, rot_y = ff.rotate_to_wind_direction(turbine_x, turbine_y, windresource.wind_directions[1]) - sorted_turbine_index = sortperm(rot_x) + # rot_x, rot_y = ff.rotate_to_wind_direction(turbine_x, turbine_y, windresource.wind_directions[1]) + # sorted_turbine_index = sortperm(rot_x) - U = ff.turbine_velocities_one_direction(rot_x, rot_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, - sorted_turbine_index, ct_models, rotor_points_y, rotor_points_z, windresource, - model_set) + # U = ff.turbine_velocities_one_direction(rot_x, rot_y, turbine_z, rotor_diameter, hub_height, turbine_yaw, + # sorted_turbine_index, ct_models, rotor_points_y, rotor_points_z, windresource, + # model_set) - @test U ≈ transpose(floris_speeds) atol=0.002 + # @test U ≈ transpose(floris_speeds) atol=0.002 - end + # end end @testset "Local Turbulence Intensity Models" begin @@ -2540,7 +2553,7 @@ using FiniteDiff include("./model_sets/model_set_1.jl") - + turbine_x = [0.0] # test no loss upstream (data from Jensen 1983) expected_velocity = wind_speed @@ -2579,7 +2592,7 @@ using FiniteDiff windresource, model_set, wind_farm_state_id=1, downwind_turbine_id=0) ≈ expected_velocity rtol=rtol end - @testset "calculate_flow_field" begin + @testset "calculate_flow_field" begin include("./model_sets/model_set_0_single_turbine.jl") @@ -2608,7 +2621,7 @@ using FiniteDiff ffvelocities = ff.calculate_flow_field(xrange, yrange, zrange, model_set, turbine_x, turbine_y, turbine_z, turbine_yaw, rotor_diameter, hub_height, ct_models, rotor_sample_points_y, rotor_sample_points_z, - wind_resource) + wind_resource) @test all(ffvelocities .== inflowuniform) @@ -2635,7 +2648,7 @@ using FiniteDiff zrange_b = data_b[:,2].*rotor_diameter[1] u0_b = zeros(length(zrange_b)) u_b = zeros(length(zrange_b)) - for i in 1:length(zrange_b) + for i in 1:length(zrange_b) u0_b[i] = ff.adjust_for_wind_shear(zrange_b[i], uh, hh, wind_shear_model.ground_height, wind_shear_model) u_b[i] = u0_b[i]*(1.0-data_b[i,1]) end @@ -2646,54 +2659,13 @@ using FiniteDiff ffvelocitiesbp2014 = ff.calculate_flow_field(xrange, yrange, zrange_b, model_set_bp2014, turbine_x, turbine_y, turbine_z, turbine_yaw, rotor_diameter, hub_height, ct_models, rotor_sample_points_y, rotor_sample_points_z, - wind_resource) + wind_resource) ffvelocitiesbp2014 = reshape(ffvelocitiesbp2014, (length(u0_b))) @test isapprox(ffvelocitiesbp2014, u_b, atol=0.1) end - - # @testset "Turbine Inflow Velocities one direction" begin - # # test based on: - # # [1] An Aero-acoustic Noise Distribution Prediction Methodology for Offshore Wind Farms - # # by Jiufa Cao, Weijun Zhu, Xinbo Wu, Tongguang Wang, and Haoran Xu - # - # rtol = 1E-6 - # - # data = readdlm("inputfiles/velocity_def_row_of_10_turbs.txt", ',', skipstart=4) - # - # rtol = 1E-6 - # - # include("./model_sets/model_set_2.jl") - # # test no loss upstream (data from Jensen 1983) - # expected_velocity = wind_speed*data[:, 2] - # - # ff.turbine_velocities_one_direction!(rotor_points_y, rotor_points_z, ms2, pd2) - # - # @test windfarmstate.turbine_inflow_velcities ≈ expected_velocity rtol=rtol - # - # end - - # @testset "Turbine powers one direction" begin - # # test based on: - # # [1] An Aero-acoustic Noise Distribution Prediction Methodology for Offshore Wind Farms - # # by Jiufa Cao, Weijun Zhu, Xinbo Wu, Tongguang Wang, and Haoran Xu - # - # rtol = 1E-6 - # - # data = readdlm("inputfiles/velocity_def_row_of_10_turbs.txt", ',', skipstart=4) - # - # rtol = 1E-6 - # - # include("./model_sets/model_set_2.jl") - # - # ff.turbine_powers_one_direction!(rotor_points_y, rotor_points_z, pd2) - # - # @test windfarmstate.turbine_generators_powers ≈ wind_speed*data[:, 2] rtol=rtol - # - # end - end @testset "IO" begin @@ -2746,11 +2718,11 @@ using FiniteDiff file_name = "./inputfiles/iea37-ex-opt3.yaml" turbine_x, turbine_y, fname_turb, fname_wr = ff.get_turb_loc_YAML(file_name) - + @test turbine_x == test_x @test turbine_y == test_y @test fname_turb == "iea37-10mw.yaml" - @test fname_wr == "iea37-windrose-cs3.yaml" + @test fname_wr == "iea37-windrose-cs3.yaml" end @@ -2827,7 +2799,7 @@ using FiniteDiff @test speeds == test_speed @test frequencies == test_freq @test ti == 0.075 - + end @testset "get_boundary_yaml" begin @@ -2839,7 +2811,7 @@ using FiniteDiff boundary_vertices_correct = [10363.8 6490.3; 9449.7 1602.2; 9387.0 1056.6; 9365.1 625.5; 9360.8 360.2; 9361.5 126.9; 9361.3 137.1; 7997.6 1457.9; 6098.3 3297.5; 8450.3 6455.3; 8505.4 6422.3; 9133.0 6127.4; 9332.8 6072.6; 9544.2 6087.1; 9739.0 6171.2; 9894.9 6316.9; 10071.8 6552.5; 10106.9 6611.1] - + @test boundary_vertices ≈ boundary_vertices_correct atol=1E-6 end @@ -2849,16 +2821,16 @@ using FiniteDiff boundary_vertices_a = [10363.8 6490.3; 9449.7 1602.2; 9387.0 1056.6; 9365.1 625.5; 9360.8 360.2; 9361.5 126.9; 9361.3 137.1; 7997.6 1457.9; 6098.3 3297.5; 8450.3 6455.3; 8505.4 6422.3; 9133.0 6127.4; 9332.8 6072.6; 9544.2 6087.1; 9739.0 6171.2; 9894.9 6316.9; 10071.8 6552.5; 10106.9 6611.1] - boundary_normals_a = [0.9829601758936983 -0.1838186405319916; 0.9934614633172962 -0.11416795042154541; 0.9987121579438882 -0.050734855622757584; - 0.9998686751666075 -0.01620593781838486; 0.9999954987444023 0.0030004151269687495; -0.9998078216567232 -0.019604074934516894; -0.6957179389375846 -0.718315076718037; - -0.6957275377423737 -0.7183057797532565; -0.8019887481131871 0.5973391397688945; 0.5138086803485797 0.8579047965820281; 0.4252760929807897 0.905063668886888; - 0.2645057513093967 0.9643841078762402; -0.0684295708121141 0.9976559496331737; -0.39636379138742883 0.9180935381958544; -0.6828023205475376 0.7306031693435896; + boundary_normals_a = [0.9829601758936983 -0.1838186405319916; 0.9934614633172962 -0.11416795042154541; 0.9987121579438882 -0.050734855622757584; + 0.9998686751666075 -0.01620593781838486; 0.9999954987444023 0.0030004151269687495; -0.9998078216567232 -0.019604074934516894; -0.6957179389375846 -0.718315076718037; + -0.6957275377423737 -0.7183057797532565; -0.8019887481131871 0.5973391397688945; 0.5138086803485797 0.8579047965820281; 0.4252760929807897 0.905063668886888; + 0.2645057513093967 0.9643841078762402; -0.0684295708121141 0.9976559496331737; -0.39636379138742883 0.9180935381958544; -0.6828023205475376 0.7306031693435896; -0.7996740386176392 0.6004343694034798; -0.8578802011411015 0.5138497450520954; 0.42552559023380465 0.9049463918134445] boundary_vertices_b = [5588.4 3791.3; 4670.7 4680.2; 7274.9 7940.8; 7369.9 7896.2; 7455.1 7784.3; 7606.5 7713.0; 7638.9 7708.4; 8297.1 7398.9] - boundary_normals_b = [-0.6957460043611584 -0.7182878931288504; -0.7813688797257963 0.6240694462926818; 0.4249708760634733 0.9052070230051488; 0.7956275395848184 0.6057861159305391; + boundary_normals_b = [-0.6957460043611584 -0.7182878931288504; -0.7813688797257963 0.6240694462926818; 0.4249708760634733 0.9052070230051488; 0.7956275395848184 0.6057861159305391; 0.4260560153872896 0.9046967844268629; 0.14056568619461773 0.9900713549359138; 0.4255255464063141 0.9049464124220882; 0.7996806883794807 -0.6004255129763556] boundary_vertices_c = [3267.1 10100.6; 4164.1 9586.6; 5749.8 9068.6; 6054.7 8925.3; 1468.5 7781.7; 107.4 9100.0] - boundary_normals_c = [0.49718026396417986 0.8676472699919642; 0.31052117525343714 0.9505664625470563; 0.42535384615162936 0.9050271297392228; 0.24194817066179167 -0.9702891747893577; + boundary_normals_c = [0.49718026396417986 0.8676472699919642; 0.31052117525343714 0.9505664625470563; 0.42535384615162936 0.9050271297392228; 0.24194817066179167 -0.9702891747893577; -0.6957228969594285 -0.7183102746351193; -0.30189947425802094 0.9533397649540959] boundary_vertices_d = [6764.9 8399.7; 4176.8 5158.6; 2047.8 7220.7] boundary_normals_d = [0.7814306689309158 -0.6239920749930895; -0.6957310325444781 -0.7183023947855072; -0.24248239299288069 0.9701558066045093]