Skip to content

Commit

Permalink
Jvolty
Browse files Browse the repository at this point in the history
  • Loading branch information
mihakralj committed Oct 14, 2024
1 parent ffed649 commit 98fc0e7
Show file tree
Hide file tree
Showing 7 changed files with 290 additions and 69 deletions.
66 changes: 0 additions & 66 deletions .github/workflows/Publish.yml
Original file line number Diff line number Diff line change
Expand Up @@ -15,72 +15,6 @@ permissions:
security-events: write # Required for CodeQL analysis and uploading SARIF results

jobs:
SonarCloud:
runs-on: ubuntu-latest
steps:
- name: Checkout repository
uses: actions/checkout@v4
with:
fetch-depth: 0

- name: Setup .NET SDK
uses: actions/setup-dotnet@v3
with:
dotnet-version: '8.x'

- name: Install JDK11 for Sonar Scanner
uses: actions/setup-java@v3
with:
java-version: '11'
distribution: 'zulu'

- name: Install dotnet-sonarscanner
run: |
dotnet tool install --global dotnet-sonarscanner
dotnet tool install JetBrains.dotCover.GlobalTool --global
dotnet tool install dotnet-coverage --global
dotnet restore
- name: SonarCloud Scanner Start
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }}
run: |
if [[ ${{ github.event_name }} == 'pull_request' ]]; then
PR_PARAMS="/d:sonar.pullrequest.key=${{ github.event.pull_request.number }} \
/d:sonar.pullrequest.branch=${{ github.head_ref }} \
/d:sonar.pullrequest.base=${{ github.base_ref }}"
elif [[ ${{ github.event_name }} == 'push' ]]; then
BRANCH_PARAMS="/d:sonar.branch.name=${{ github.ref_name }}"
else
BRANCH_PARAMS="/d:sonar.branch.name=${{ github.ref_name }}"
fi
dotnet sonarscanner begin \
/k:"mihakralj_QuanTAlib" \
/o:"mihakralj" \
/d:sonar.token="${{ secrets.SONAR_TOKEN }}" \
/d:sonar.host.url="https://sonarcloud.io" \
/d:sonar.cs.dotcover.reportsPaths=dotcover* \
/d:sonar.scanner.scanAll=false \
/d:sonar.scm.provider=git \
$PR_PARAMS $BRANCH_PARAMS
- name: Build
run: |
dotnet build --no-restore --configuration Debug
dotnet build ./lib/quantalib.csproj --configuration Release --nologo
dotnet build ./quantower/Averages/_Averages.csproj --configuration Release --nologo
dotnet build ./quantower/Statistics/_Statistics.csproj --configuration Release --nologo
dotnet build ./quantower/Volatility/_Volatility.csproj --configuration Release --nologo
dotnet build ./SyntheticVendor/SyntheticVendor.csproj --configuration Release --nologo
dotnet dotcover test Tests/Tests.csproj --dcReportType=HTML --dcoutput=./dotcover.html
- name: SonarCloud Scanner End
env:
SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }}
run: dotnet sonarscanner end /d:sonar.token="${{ secrets.SONAR_TOKEN }}"

Code_Coverage:
runs-on: ubuntu-latest
steps:
Expand Down
3 changes: 1 addition & 2 deletions Tests/test_iTBar.cs
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,6 @@
using System.Diagnostics.CodeAnalysis;
using System.Security.Cryptography;

#pragma warning disable S1944, S2053, S2222, S2259, S2583, S2589, S3329, S3655, S3900, S3949, S3966, S4158, S4347, S5773, S6781

namespace QuanTAlib;

/// <summary>
Expand All @@ -27,6 +25,7 @@ public BarIndicatorTests()
private static readonly ITValue[] indicators = new ITValue[]
{
new Atr(period: 14),
new Jvolty(period: 14)
// Add other TBar-based indicators here
};

Expand Down
2 changes: 2 additions & 0 deletions Tests/test_quantower.cs
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,8 @@ public class QuantowerTests

// Volatility Indicators
[Fact] public void Atr() => TestIndicator<AtrIndicator>("atr");
[Fact] public void Jvolty() => TestIndicator<JvoltyIndicator>("jvolty");

[Fact] public void Historical() => TestIndicator<HistoricalIndicator>("historical");
[Fact] public void Realized() => TestIndicator<RealizedIndicator>("realized");
[Fact] public void Rvi() => TestIndicator<RviIndicator>("rvi");
Expand Down
104 changes: 104 additions & 0 deletions Tests/test_updates_volatility.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,104 @@
using Xunit;
using System.Security.Cryptography;

namespace QuanTAlib.Tests;

public class VolatilityUpdateTests
{
private readonly RandomNumberGenerator rng = RandomNumberGenerator.Create();
private const int RandomUpdates = 100;
private const double ReferenceValue = 100.0;
private const int precision = 8;

private double GetRandomDouble()
{
byte[] bytes = new byte[8];
rng.GetBytes(bytes);
return (double)BitConverter.ToUInt64(bytes, 0) / ulong.MaxValue * 200 - 100; // Range: -100 to 100
}

private TBar GetRandomBar(bool IsNew)
{
double open = GetRandomDouble();
double high = open + Math.Abs(GetRandomDouble());
double low = open - Math.Abs(GetRandomDouble());
double close = low + (high - low) * GetRandomDouble();
return new TBar(DateTime.Now, open, high, low, close, 1000, IsNew);
}

[Fact]
public void Atr_Update()
{
var indicator = new Atr(period: 14);
TBar r = GetRandomBar(true);
double initialValue = indicator.Calc(r);

for (int i = 0; i < RandomUpdates; i++)
{
indicator.Calc(GetRandomBar(IsNew: false));
}
double finalValue = indicator.Calc(new TBar(r.Time, r.Open, r.High, r.Low, r.Close, r.Volume, IsNew: false));

Assert.Equal(initialValue, finalValue, precision);
}

[Fact]
public void Historical_Update()
{
var indicator = new Historical(period: 14);
double initialValue = indicator.Calc(new TBar(DateTime.Now, ReferenceValue, ReferenceValue, ReferenceValue, ReferenceValue, 1000, IsNew: true));

for (int i = 0; i < RandomUpdates; i++)
{
indicator.Calc(GetRandomBar(false));
}
double finalValue = indicator.Calc(new TBar(DateTime.Now, ReferenceValue, ReferenceValue, ReferenceValue, ReferenceValue, 1000, IsNew: false));

Assert.Equal(initialValue, finalValue, precision);
}

[Fact]
public void Jvolty_Update()
{
var indicator = new Jvolty(period: 14);
double initialValue = indicator.Calc(new TBar(DateTime.Now, ReferenceValue, ReferenceValue, ReferenceValue, ReferenceValue, 1000, IsNew: true));

for (int i = 0; i < RandomUpdates; i++)
{
indicator.Calc(GetRandomBar(false));
}
double finalValue = indicator.Calc(new TBar(DateTime.Now, ReferenceValue, ReferenceValue, ReferenceValue, ReferenceValue, 1000, IsNew: false));

Assert.Equal(initialValue, finalValue, precision);
}

[Fact]
public void Realized_Update()
{
var indicator = new Realized(period: 14);
double initialValue = indicator.Calc(new TBar(DateTime.Now, ReferenceValue, ReferenceValue, ReferenceValue, ReferenceValue, 1000, IsNew: true));

for (int i = 0; i < RandomUpdates; i++)
{
indicator.Calc(GetRandomBar(false));
}
double finalValue = indicator.Calc(new TBar(DateTime.Now, ReferenceValue, ReferenceValue, ReferenceValue, ReferenceValue, 1000, IsNew: false));

Assert.Equal(initialValue, finalValue, precision);
}

[Fact]
public void Rvi_Update()
{
var indicator = new Rvi(period: 14);
double initialValue = indicator.Calc(new TBar(DateTime.Now, ReferenceValue, ReferenceValue, ReferenceValue, ReferenceValue, 1000, IsNew: true));

for (int i = 0; i < RandomUpdates; i++)
{
indicator.Calc(GetRandomBar(false));
}
double finalValue = indicator.Calc(new TBar(DateTime.Now, ReferenceValue, ReferenceValue, ReferenceValue, ReferenceValue, 1000, IsNew: false));

Assert.Equal(initialValue, finalValue, precision);
}
}
141 changes: 141 additions & 0 deletions lib/volatility/Jvolty.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,141 @@
/// <summary>
/// Represents a Jurik Volatility (Jvolty) calculator, a measure of market volatility based on Jurik Moving Average (JMA) concepts.
/// </summary>

namespace QuanTAlib;

public class Jvolty : AbstractBase
{
private readonly int _period;
private readonly CircularBuffer _values;
private readonly CircularBuffer _voltyShort;
private readonly CircularBuffer _vsumBuff;
private readonly CircularBuffer _avoltyBuff;

private double _len1;
private double _pow1;
private double _upperBand;
private double _lowerBand;
private double _p_upperBand;
private double _p_lowerBand;

/// <summary>
/// Initializes a new instance of the Jvolty class with the specified parameters.
/// </summary>
/// <param name="period">The period over which to calculate the Jvolty.</param>
/// <param name="phase">The phase parameter for the JMA-style calculation.</param>
/// <param name="vshort">The short-term volatility period.</param>
/// <exception cref="ArgumentOutOfRangeException">
/// Thrown when period is less than 1.
/// </exception>
public Jvolty(int period, int vshort = 10)
{
if (period < 1)
{
throw new ArgumentOutOfRangeException(nameof(period), "Period must be greater than or equal to 1.");
}
_period = period;
int _vlong = 65;

_values = new CircularBuffer(period);
_voltyShort = new CircularBuffer(vshort);
_vsumBuff = new CircularBuffer(_vlong);
_avoltyBuff = new CircularBuffer(2);

WarmupPeriod = period * 2;
Name = $"JVOLTY({period},{vshort})";
}

/// <summary>
/// Initializes a new instance of the Jvolty class with the specified source and parameters.
/// </summary>
/// <param name="source">The source object to subscribe to for bar updates.</param>
/// <param name="period">The period over which to calculate the Jvolty.</param>
/// <param name="phase">The phase parameter for the JMA-style calculation.</param>
/// <param name="vshort">The short-term volatility period.</param>
public Jvolty(object source, int period, int vshort = 10) : this(period, vshort)
{
var pubEvent = source.GetType().GetEvent("Pub");
pubEvent?.AddEventHandler(source, new BarSignal(Sub));
}

/// <summary>
/// Initializes the Jvolty instance by setting up the initial state.
/// </summary>
public override void Init()
{
base.Init();
_upperBand = _lowerBand = 0.0;
_p_upperBand = _p_lowerBand = 0.0;
_len1 = Math.Max((Math.Log(Math.Sqrt(_period - 1)) / Math.Log(2.0)) + 2.0, 0);
_pow1 = Math.Max(_len1 - 2.0, 0.5);
_avoltyBuff.Clear();
_avoltyBuff.Add(0, true);
_avoltyBuff.Add(0, true);
}

/// <summary>
/// Manages the state of the Jvolty instance based on whether a new bar is being processed.
/// </summary>
/// <param name="isNew">Indicates whether the current input is a new bar.</param>
protected override void ManageState(bool isNew)
{
if (isNew)
{
_index++;
_p_upperBand = _upperBand;
_p_lowerBand = _lowerBand;
}
else
{
_upperBand = _p_upperBand;
_lowerBand = _p_lowerBand;
}
}

/// <summary>
/// Performs the Jvolty calculation for the current bar.
/// </summary>
/// <returns>
/// The calculated Jvolty value for the current bar.
/// </returns>
protected override double Calculation()
{
ManageState(BarInput.IsNew);

_values.Add(BarInput.Close, BarInput.IsNew);

if (_index == 1)
{
return 0;
}

double hprice = _values.Max();
double lprice = _values.Min();

double del1 = hprice - _upperBand;
double del2 = lprice - _lowerBand;
double volty = Math.Max(Math.Abs(del1), Math.Abs(del2));

_voltyShort.Add(volty, BarInput.IsNew);
double vsum = _vsumBuff.Newest() + 0.1 * (volty - _voltyShort.Oldest());
_vsumBuff.Add(vsum, BarInput.IsNew);

double prevAvolty = _avoltyBuff.Newest();
double avolty = prevAvolty + 2.0 / (Math.Max(4.0 * _period, 30) + 1.0) * (vsum - prevAvolty);
_avoltyBuff.Add(avolty, BarInput.IsNew);

double dVolty = (avolty > 0) ? volty / avolty : 0;
dVolty = Math.Min(Math.Max(dVolty, 1.0), Math.Pow(_len1, 1.0 / _pow1));

double pow2 = Math.Pow(dVolty, _pow1);
double len2 = Math.Sqrt(0.5 * (_period - 1)) * _len1;
double Kv = Math.Pow(len2 / (len2 + 1), Math.Sqrt(pow2));

_upperBand = (del1 > 0) ? hprice : hprice - (Kv * del1);
_lowerBand = (del2 < 0) ? lprice : lprice - (Kv * del2);

IsHot = _index >= WarmupPeriod;
return volty;
}
}
2 changes: 1 addition & 1 deletion lib/volatility/todo.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@

## Single Value Input (Typically Closing Prices)

- Jurik Volatility (Volty)
- **Jurik Volatility (Volty)**
- **Standard Deviation**
- **Relative Volatility Index (RVI)**
- Ulcer Index
Expand Down
41 changes: 41 additions & 0 deletions quantower/Volatility/JvoltyIndicator.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
using System.Drawing;
using TradingPlatform.BusinessLayer;

namespace QuanTAlib;

public class JvoltyIndicator : Indicator, IWatchlistIndicator
{
[InputParameter("Periods", sortIndex: 1, 1, 2000, 1, 0)]
public int Periods { get; set; } = 20;

private Jvolty? jvolty;
protected LineSeries? JvoltySeries;
public static int MinHistoryDepths => 2;
int IWatchlistIndicator.MinHistoryDepths => MinHistoryDepths;

public JvoltyIndicator()
{
Name = "JVOLTY - Mark Jurik's Volatility";
Description = "Measures market volatility according to Mark Jurik.";
SeparateWindow = true;

JvoltySeries = new("JVOLTY", Color.Blue, 2, LineStyle.Solid);
AddLineSeries(JvoltySeries);
}

protected override void OnInit()
{
jvolty = new (Periods);
base.OnInit();
}

protected override void OnUpdate(UpdateArgs args)
{
TBar input = IndicatorExtensions.GetInputBar(this, args);
TValue result = jvolty!.Calc(input);

JvoltySeries!.SetValue(result.Value);
}

public override string ShortName => $"JVOLTY ({Periods})";
}

0 comments on commit 98fc0e7

Please sign in to comment.