diff --git a/.gitignore b/.gitignore index d001e67..53bb2a6 100644 --- a/.gitignore +++ b/.gitignore @@ -262,3 +262,4 @@ __pycache__/ /PTIRelianceLib/build/* _site/ nuget/ +/RelianceCLI/build/ diff --git a/PTIRelianceLib.Tests/IO/Internal/HidDeviceInfoTests.cs b/PTIRelianceLib.Tests/IO/Internal/HidDeviceInfoTests.cs new file mode 100644 index 0000000..5ee5e1d --- /dev/null +++ b/PTIRelianceLib.Tests/IO/Internal/HidDeviceInfoTests.cs @@ -0,0 +1,33 @@ +using System; +using System.Linq; +using Xunit; + +namespace PTIRelianceLib.Tests.IO.Internal +{ + using PTIRelianceLib.IO.Internal; + + public class HidDeviceInfoTests + { + [Fact] + public void TestToString() + { + var devinfo = new HidDeviceInfo + { + VendorId = 0x1234, + ProductId = 0x5678, + ManufacturerString = "Company", + ProductString = "Product", + Path = "/some/device&1234_5678/4325EACF" + }; + + var str = devinfo.ToString(); + Assert.NotNull(str); + + Assert.Contains(devinfo.ManufacturerString, str); + Assert.Contains(devinfo.ProductString, str); + Assert.Contains(devinfo.VendorId.ToString("X4"), str); + Assert.Contains(devinfo.ProductId.ToString("X4"), str); + } + } +} + diff --git a/PTIRelianceLib.Tests/IO/StructuredReaderTests.cs b/PTIRelianceLib.Tests/IO/StructuredReaderTests.cs new file mode 100644 index 0000000..a24f43a --- /dev/null +++ b/PTIRelianceLib.Tests/IO/StructuredReaderTests.cs @@ -0,0 +1,92 @@ +using System; +using System.Linq; +using Xunit; + +namespace PTIRelianceLib.Tests.IO +{ + using PTIRelianceLib.IO; + using PTIRelianceLib.IO.Internal; + using PTIRelianceLib.Transport; + + public class StructuredReaderTests + { + private static readonly object MTestLock = new object(); + + private readonly HidDeviceConfig _mConfig; + private readonly FakeNativeMethods _mNativeMock; + private readonly HidPort _mPort; + public const int VendorId = 0x0425; + public const int ProductId = 0x8147; + + public StructuredReaderTests() + { + _mNativeMock = new FakeNativeMethods(); + + // Reliance will "always" use report lengths of 34 bytes + _mConfig = new HidDeviceConfig + { + VendorId = VendorId, + ProductId = ProductId, + InReportLength = 34, + OutReportLength = 34, + InReportId = 2, + OutReportId = 1, + NativeHid = _mNativeMock + }; + + _mPort = new HidPort(_mConfig); + } + + [Fact] + public void TestCtor() + { + lock (MTestLock) + { + _mNativeMock.GetNextResponse = (d) => GenerateHidData(0xAA); + new StructuredReader(_mPort); + } + } + + + [Fact] + public void TestReadNak() + { + lock (MTestLock) + { + _mNativeMock.GetNextResponse = (d) => GenerateHidData(0xAC); + var reader = new StructuredReader(_mPort); + var resp = reader.Read(1, 2, 3); + Assert.True(resp.IsEmpty); + } + } + + [Fact] + public void TestReadAck() + { + lock (MTestLock) + { + _mNativeMock.GetNextResponse = (d) => GenerateHidData(0xAA, 1, 2, 3); + var reader = new StructuredReader(_mPort); + var resp = reader.Read(3, 1, 2, 3); + Assert.False(resp.IsEmpty); + } + } + + /// + /// Generates a valid inreport + /// + /// Payload to pack + /// + private byte[] GenerateHidData(params byte[] payload) + { + var packet = new ReliancePacket(payload); + payload = packet.Package().GetBytes(); + + var buff = new byte[_mConfig.InReportLength]; + buff[0] = _mConfig.InReportId; + buff[1] = (byte) payload.Length; + Array.Copy(payload, 0, buff, 2, Math.Min(payload.Length, buff.Length - 2)); + return buff; + } + } +} \ No newline at end of file diff --git a/PTIRelianceLib.Tests/Imaging/BasePrintLogoTests.cs b/PTIRelianceLib.Tests/Imaging/BasePrintLogoTests.cs index 75e77c2..e4762c0 100644 --- a/PTIRelianceLib.Tests/Imaging/BasePrintLogoTests.cs +++ b/PTIRelianceLib.Tests/Imaging/BasePrintLogoTests.cs @@ -1,37 +1,59 @@ -using PTIRelianceLib.Imaging; +using System; using Xunit; namespace PTIRelianceLib.Tests.Imaging { - using System.ComponentModel; + using Properties; + using PTIRelianceLib.Imaging; public class BasePrintLogoTests { - [Fact()] - [Category("BMP")] - public void ApplyColorInversionTest() + [Fact] + public void TestCtor() { - // Input are expected are provided as resources, dithered is what - // we are testing + var bitmap = BinaryFile.From(Resources.black_bitmap); + var logo = new BasePrintLogo(bitmap); - var input = BinaryFile.From(Properties.Resources.white_bitmap); + // Dimensions should be unchanged, 48x48 + Assert.Equal(48, logo.IdealHeight); + Assert.Equal(48, logo.IdealWidth); - var logo = new BasePrintLogo(input); + // Max width/height were not specified, should be 0 + Assert.Equal(0, logo.MaxHeight); + Assert.Equal(0, logo.MaxWidth); - Assert.False(logo.IsInverted); + // Valid data was passed in, ImageData should be valid too + Assert.NotNull(logo.ImageData); + } + + [Fact] + public void TestCtorNullBitmap() + { + Assert.Throws(() => new BasePrintLogo(null)); + } + + [Fact] + public void TestApplyDithering() + { + var bitmap = BinaryFile.From(Resources.gray_bitmap); + var logo = new BasePrintLogo(bitmap); - logo.ApplyColorInversion(); + var startw = logo.IdealWidth; + var starth = logo.IdealHeight; + var predither = Crc32.ComputeChecksum(logo.ToBuffer()); - var inverted = logo.ImageData; - var expected = new BasePrintLogo(BinaryFile.From(Properties.Resources.black_bitmap)).ImageData; + // If this fails, someone dispose of the bitmap along the way. + // Look for "using" statements to fix + logo.ApplyDithering(DitherAlgorithms.Atkinson); - // White should ivnert to black - Assert.True(ImageTestHelpers.CompareCrc32(expected, inverted)); - Assert.True(logo.IsInverted); + // Dimensions should be unchanged, 48x48 + Assert.Equal(starth, logo.IdealHeight); + Assert.Equal(startw, logo.IdealWidth); - // Flip back to white, test that the inversion flag is cleared - logo.ApplyColorInversion(); - Assert.False(logo.IsInverted); + // Valid data was passed in, ImageData should be valid too + Assert.NotNull(logo.ImageData); + var postdither = Crc32.ComputeChecksum(logo.ToBuffer()); + Assert.NotEqual(predither, postdither); } } } \ No newline at end of file diff --git a/PTIRelianceLib.Tests/Imaging/DitherFactoryTests.cs b/PTIRelianceLib.Tests/Imaging/DitherFactoryTests.cs index b0653b5..ac3c76d 100644 --- a/PTIRelianceLib.Tests/Imaging/DitherFactoryTests.cs +++ b/PTIRelianceLib.Tests/Imaging/DitherFactoryTests.cs @@ -52,7 +52,14 @@ public void GetDithererFloydSteinbergFact() Assert.True(ImageTestHelpers.CompareCrc32(expected, dithered)); } - // TODO test false floyd steinberg + [Fact()] + public void GetDithererFloydSteinbergFalseFact() + { + var input = new BasePrintLogo(BinaryFile.From(Properties.Resources.gray_bitmap)).ImageData; + var expected = new BasePrintLogo(BinaryFile.From(Properties.Resources.gray_floydsteinbergsfalse)).ImageData; + var dithered = DitherFactory.GetDitherer(DitherAlgorithms.FloydSteinbergFalse).GenerateDithered(input); + Assert.True(ImageTestHelpers.CompareCrc32(expected, dithered)); + } [Fact()] public void GetDithererJarvisJudiceNinkeFact() diff --git a/PTIRelianceLib.Tests/Imaging/ImageExtTests.cs b/PTIRelianceLib.Tests/Imaging/ImageExtTests.cs index 3951d68..7596891 100644 --- a/PTIRelianceLib.Tests/Imaging/ImageExtTests.cs +++ b/PTIRelianceLib.Tests/Imaging/ImageExtTests.cs @@ -162,6 +162,12 @@ public void BitmapToLogoBufferSimpleFact() Assert.Equal(expectedBuff, actualBuff); } + [Fact] + public void TestNullBitmapToBuffer() + { + Assert.Empty(ImageExt.ToBuffer(null)); + } + /// /// Returns a list of type T with value repeat count times /// diff --git a/PTIRelianceLib.Tests/Imaging/LogoStorageConfigTests.cs b/PTIRelianceLib.Tests/Imaging/LogoStorageConfigTests.cs new file mode 100644 index 0000000..11b9954 --- /dev/null +++ b/PTIRelianceLib.Tests/Imaging/LogoStorageConfigTests.cs @@ -0,0 +1,39 @@ +using Xunit; + +namespace PTIRelianceLib.Tests.Imaging +{ + using PTIRelianceLib.Imaging; + + public class LogoStorageConfigTests + { + [Fact] + public void FailIfDefaultsChange() + { + var config = LogoStorageConfig.Default; + Assert.Equal(640, config.MaxPixelWidth); + Assert.Equal(127, config.Threshold); + Assert.Equal(DitherAlgorithms.None, config.Algorithm); + + // Make sure properties stay editable + config.MaxPixelWidth = 0; + config.Threshold = 0; + config.Algorithm = DitherAlgorithms.Burkes; + } + + [Fact] + public void FailIfDefaultsAreReadonly() + { + var config = LogoStorageConfig.Default; + + // Make sure properties stay editable + config.MaxPixelWidth = 0; + config.Threshold = 0; + config.Algorithm = DitherAlgorithms.Burkes; + + + Assert.Equal(0, config.MaxPixelWidth); + Assert.Equal(0, config.Threshold); + Assert.Equal(DitherAlgorithms.Burkes, config.Algorithm); + } + } +} diff --git a/PTIRelianceLib.Tests/Internal/ExtensionsTests.cs b/PTIRelianceLib.Tests/Internal/ExtensionsTests.cs new file mode 100644 index 0000000..4a4e28a --- /dev/null +++ b/PTIRelianceLib.Tests/Internal/ExtensionsTests.cs @@ -0,0 +1,33 @@ +using Xunit; + +namespace PTIRelianceLib.Tests.Internal +{ + using System.Collections.Generic; + + public class ExtensionsTests + { + [Fact] + public void ToUshortBE() + { + var data = new Dictionary + { + {new byte[] {0, 0}, 0}, + {new byte[] {1, 0}, 1}, + {new byte[] {2, 1}, 258}, + {new byte[] {3, 2}, 515}, + {new byte[] {4, 3}, 772}, + {new byte[] {255, 255}, 0xFFFF} + }; + + foreach (var kv in data) + { + var actual1 = kv.Key.ToUshortBE(); + Assert.Equal(kv.Value, actual1); + + // Run the reverse test on the same data + var actual2 = actual1.ToBytesBE(); + Assert.Equal(kv.Key, actual2); + } + } + } +} diff --git a/PTIRelianceLib.Tests/Internal/FixedArrayTests.cs b/PTIRelianceLib.Tests/Internal/FixedArrayTests.cs new file mode 100644 index 0000000..a2149da --- /dev/null +++ b/PTIRelianceLib.Tests/Internal/FixedArrayTests.cs @@ -0,0 +1,59 @@ +using System; +using System.Linq; +using Xunit; + +namespace PTIRelianceLib.Tests.Internal +{ + public class FixedArrayTests + { + [Fact] + public void TestCtor() + { + var farr = new FixedArray(10); + Assert.Equal(10, farr.Size); + Assert.Equal(0, farr.Count); + Assert.True(string.IsNullOrEmpty(farr.Content)); + } + + [Fact] + public void TestSetData() + { + var farr = new FixedArray(3); + + farr.SetData( + BinaryFile.From(new byte[0]), + BinaryFile.From(new byte[1]), + BinaryFile.From(new byte[2]), + BinaryFile.From(new byte[3]) // This one should be silently discarded + ); + + Assert.Equal(farr.Count, farr.Size); + + var data = farr.GetData(); + Assert.Equal(data.Length, farr.Size); + + // Test that each length is what we expect + var len = 0; + foreach (var bf in data) + { + Assert.Equal(len++, bf.Length); + } + Assert.False(string.IsNullOrEmpty(farr.Content)); + } + + [Fact] + public void TestIndexer() + { + var farr = new FixedArray(3); + var obj1 = new object(); + var obj2 = new object(); + var obj3 = new object(); + + farr.SetData(obj1, obj2, obj3); + + Assert.Equal(obj1, farr[0]); + Assert.Equal(obj2, farr[1]); + Assert.Equal(obj3, farr[2]); + } + } +} diff --git a/PTIRelianceLib.Tests/MutableReliancePacket.cs b/PTIRelianceLib.Tests/MutableReliancePacket.cs index a3a3e51..d84bb3c 100644 --- a/PTIRelianceLib.Tests/MutableReliancePacket.cs +++ b/PTIRelianceLib.Tests/MutableReliancePacket.cs @@ -13,6 +13,10 @@ namespace PTIRelianceLib.Tests internal class MutableReliancePacket : ReliancePacket { + public MutableReliancePacket() + { + } + internal MutableReliancePacket(byte[] data) : base(data) { } diff --git a/PTIRelianceLib.Tests/Properties/Resources.Designer.cs b/PTIRelianceLib.Tests/Properties/Resources.Designer.cs index 9d8e92c..a5e2f71 100644 --- a/PTIRelianceLib.Tests/Properties/Resources.Designer.cs +++ b/PTIRelianceLib.Tests/Properties/Resources.Designer.cs @@ -120,6 +120,16 @@ internal static byte[] gray_floydsteinbergs { } } + /// + /// Looks up a localized resource of type System.Byte[]. + /// + internal static byte[] gray_floydsteinbergsfalse { + get { + object obj = ResourceManager.GetObject("gray_floydsteinbergsfalse", resourceCulture); + return ((byte[])(obj)); + } + } + /// /// Looks up a localized resource of type System.Byte[]. /// @@ -190,6 +200,16 @@ internal static byte[] red_bitmap { } } + /// + /// Looks up a localized resource of type System.Byte[]. + /// + internal static byte[] telemetry_v3 { + get { + object obj = ResourceManager.GetObject("telemetry_v3", resourceCulture); + return ((byte[])(obj)); + } + } + /// /// Looks up a localized resource of type System.Byte[]. /// diff --git a/PTIRelianceLib.Tests/Properties/Resources.resx b/PTIRelianceLib.Tests/Properties/Resources.resx index d1f1e79..a704337 100644 --- a/PTIRelianceLib.Tests/Properties/Resources.resx +++ b/PTIRelianceLib.Tests/Properties/Resources.resx @@ -136,6 +136,9 @@ ..\Resources\gray_floydsteinbergs.bmp;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + ..\Resources\gray_floydsteinbergsfalse.bmp;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + ..\Resources\gray_jjn.bmp;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 @@ -157,6 +160,9 @@ ..\Resources\red_bitmap.bmp;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + + ..\Resources\telemetry.v3.bin;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 + ..\Resources\white_bitmap.bmp;System.Byte[], mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 diff --git a/PTIRelianceLib.Tests/Resources/gray_floydsteinbergsfalse.bmp b/PTIRelianceLib.Tests/Resources/gray_floydsteinbergsfalse.bmp new file mode 100644 index 0000000..466f488 Binary files /dev/null and b/PTIRelianceLib.Tests/Resources/gray_floydsteinbergsfalse.bmp differ diff --git a/PTIRelianceLib.Tests/Telemetry/LifetimeTelemetryParserTests.cs b/PTIRelianceLib.Tests/Telemetry/LifetimeTelemetryParserTests.cs new file mode 100644 index 0000000..13596c5 --- /dev/null +++ b/PTIRelianceLib.Tests/Telemetry/LifetimeTelemetryParserTests.cs @@ -0,0 +1,106 @@ +using System; +using Xunit; + +namespace PTIRelianceLib.Tests.Telemetry +{ + using PTIRelianceLib.Telemetry; + using PTIRelianceLib.Transport; + + public class LifetimeTelemetryParserTests + { + [Fact] + public void TestIsRegistered() + { + PacketParserFactory.Instance.Create(); + } + + [Fact] + public void TestNullPacket() + { + // Test the null object is returned + var parser = PacketParserFactory.Instance.Create(); + var tel = parser.Parse(null); + Assert.Null(tel); + } + + [Fact] + public void TestShortPacket() + { + // Test the null object is returned + var parser = PacketParserFactory.Instance.Create(); + var tel = parser.Parse(new ReliancePacket(new byte[7])); + Assert.Null(tel); + } + + [Fact] + public void TestMinimumPacket() + { + // Test the null object is returned + var parser = PacketParserFactory.Instance.Create(); + var tel = parser.Parse(new ReliancePacket(new byte[8])); + Assert.Null(tel); + } + + [Fact] + public void TestZeroedPacket() + { + // Test the null object is returned when struct size field is zero + var parser = PacketParserFactory.Instance.Create(); + var tel = parser.Parse(new ReliancePacket(new byte[112])); // arbitrarily large + Assert.Null(tel); + } + + [Fact] + public void TestMaxedPacket() + { + // Test the null object is returned when struct size field is zero + var parser = PacketParserFactory.Instance.Create(); + var data = new byte[112]; // arbitrary size + Array.Fill(data, 0xFF); // size set to INT.MAX + + var tel = parser.Parse(new ReliancePacket(data)); + Assert.Null(tel); + } + + [Fact] + public void TestInValidLengthPacket() + { + // Test that a valid size field can be processed + var parser = PacketParserFactory.Instance.Create(); + var data = new byte[9]; // less than decalred size (10) + data[4] = 10; // size set to 10 + + var tel = parser.Parse(new ReliancePacket(data)); + Assert.Null(tel); + } + + [Fact] + public void TestSerializeEmpty() + { + // Serialize is not implement, should be empty + var tel = new LifetimeTelemetry(); + Assert.Empty(tel.Serialize()); + } + + [Fact] + public void TestValidLengthPacket() + { + // Test that a valid size field can be processed + var parser = PacketParserFactory.Instance.Create(); + var data = new byte[240]; // arbitrary size + data[4] = 10; // size set to 10 + + var tel = parser.Parse(new ReliancePacket(data)); + Assert.NotNull(tel); + + Assert.Equal(0, tel.PowerUpCount); + Assert.Equal(0, tel.ResetCmdCount); + + tel.PowerUpCount++; + tel.ResetCmdCount++; + + Assert.Equal(1, tel.PowerUpCount); + Assert.Equal(1, tel.ResetCmdCount); + } + } +} diff --git a/PTIRelianceLib.Tests/Telemetry/PowerupTelemetryParserTests.cs b/PTIRelianceLib.Tests/Telemetry/PowerupTelemetryParserTests.cs new file mode 100644 index 0000000..370de14 --- /dev/null +++ b/PTIRelianceLib.Tests/Telemetry/PowerupTelemetryParserTests.cs @@ -0,0 +1,96 @@ +using Xunit; + +namespace PTIRelianceLib.Tests.Telemetry +{ + using System; + using System.Linq; + using System.Security.Cryptography; + using PTIRelianceLib.Telemetry; + using PTIRelianceLib.Transport; + + public class PowerupTelemetryParserTests + { + [Fact] + public void TestIsRegistered() + { + PacketParserFactory.Instance.Create(); + } + + [Fact] + public void TestNullPacket() + { + // Test the null object is returned + var parser = PacketParserFactory.Instance.Create(); + var tel = parser.Parse(null); + Assert.Null(tel); + } + + [Fact] + public void TestValidLengthPacket() + { + // Test that a valid size field can be processed + var parser = PacketParserFactory.Instance.Create(); + var data = new byte[240]; // arbitrary size + data[4] = 10; // size set to 10 + + var tel = parser.Parse(new ReliancePacket(data)); + Assert.NotNull(tel); + } + + [Fact] + public void TestSerializeEmpty() + { + // Serialize is not implement, should be empty + var tel = new PowerupTelemetry(); + Assert.Empty(tel.Serialize()); + } + + [Fact] + public void TestRev3Struct() + { + // Test that a valid response payload can be parsed + var parser = PacketParserFactory.Instance.Create(); + var data = Properties.Resources.telemetry_v3; + var tel = parser.Parse(new ReliancePacket(data)); + Assert.NotNull(tel); + + Assert.Equal(192, tel.AvgCutTime); + Assert.Equal(252, tel.AvgTimePresented); + Assert.Equal(0, tel.Button); + Assert.Equal(0, tel.CriticalErrorCount); + Assert.Equal(0, tel.CutterCount); + Assert.Equal(1, tel.HighErrorCount); + Assert.Equal(0, tel.JammedCount); + Assert.Equal(0, tel.OverheatedCount); + Assert.Equal(42688, tel.PaperMovedCount); + Assert.Equal(1, tel.PaperOut); + Assert.Equal(0, tel.PaperOutDuringPrint); + Assert.Equal(0, tel.PlatenOpenCount); + Assert.Equal(3, tel.StructRevision); + Assert.Equal(172, tel.StructSize); + + Assert.Equal(9, tel.TicketLengthLog.Size); + Assert.Equal(9, tel.TicketLengthLog.Count); + + var lens = new[] {28, 1, 0, 0, 0, 0, 1, 0, 1}; + var lenGroups = Enum.GetValues(typeof(TicketLengthGroups)).Cast().ToArray(); + for (var i = 0; i < lens.Length; ++i) + { + Assert.Equal(lens[i], tel.TicketLengthLog[i]); + Assert.Equal(lens[i], tel.TicketCountsByLength(lenGroups[i])); + } + + var times = new[] {5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; + var timeGroups = Enum.GetValues(typeof(TicketPullTimeGroups)).Cast().ToArray(); + for (var i = 0; i < times.Length; ++i) + { + Assert.Equal(times[i], tel.TicketPullTimeLog[i]); + Assert.Equal(times[i], tel.TicketCountByTimeToPull(timeGroups[i])); + } + + Assert.Equal(31, tel.TicketsEjected); + Assert.Equal(5, tel.TicketsPulled); + Assert.Equal(0, tel.TicketsRetracted); + } + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Configuration/RElConfigUpdater.cs b/PTIRelianceLib/Configuration/RElConfigUpdater.cs index 615cd20..4395be4 100644 --- a/PTIRelianceLib/Configuration/RElConfigUpdater.cs +++ b/PTIRelianceLib/Configuration/RElConfigUpdater.cs @@ -140,27 +140,35 @@ public ReturnCodes WriteConfiguration(BinaryFile file) public BinaryFile ReadConfiguration() { var config = new RELConfig(); - - var serial = GetConfig(_mPrinter, RelianceCommands.GetSerialConfig); - if (serial == null) + if (!FillSerialConfig(ref config)) { - // Read failure return BinaryFile.From(new byte[0]); } - config.BaudRate = serial.BaudRate; - config.Handshake = serial.Handshake; - config.Databits = serial.Databits; - config.Stopbits = serial.Stopbits; - config.Parity = serial.Parity; + FillGeneralConfig(ref config); + FillFontConfig(ref config); + FillBezelConfig(ref config); + using (var stream = new MemoryStream()) + { + config.Save(stream); + return BinaryFile.From(stream.GetBuffer()); + } + } + + /// + /// Reads in and fill config with all general parameters + /// + /// Configuration to fill + private void FillGeneralConfig(ref RELConfig config) + { config.Quality = (ReliancePrintQuality)(GetConfig(_mPrinter, RelianceCommands.GetPrintQuality) ?? 0); config.RetractEnabled = GetConfig(_mPrinter, RelianceCommands.GetRetractEnabled) == 1; config.Ejector = (RelianceEjectorMode)(GetConfig(_mPrinter, RelianceCommands.GetEjectorMode) ?? 0); config.TicketTimeout = GetConfig(_mPrinter, RelianceCommands.GetTicketTimeoutPeriod) ?? 5; - config.TicketTimeoutAction = (TicketTimeoutAction) (GetConfig(_mPrinter, RelianceCommands.GetTimeoutAction) ?? 0); - config.NewTicketAction = (NewTicketAction) (GetConfig(_mPrinter, RelianceCommands.GetNewTicketAction) ?? 1); + config.TicketTimeoutAction = (TicketTimeoutAction)(GetConfig(_mPrinter, RelianceCommands.GetTimeoutAction) ?? 0); + config.NewTicketAction = (NewTicketAction)(GetConfig(_mPrinter, RelianceCommands.GetNewTicketAction) ?? 1); config.PresentLength = GetConfig(_mPrinter, RelianceCommands.GetPresentLen) ?? 48; config.CRLFEnabled = GetConfig(_mPrinter, RelianceCommands.GetCRLFConf) == 1; config.PrintDensity = GetConfig(_mPrinter, RelianceCommands.GetPrintDensity) ?? 100; @@ -168,11 +176,45 @@ public BinaryFile ReadConfiguration() config.AutocutTimeout = GetConfig(_mPrinter, RelianceCommands.AutocutSub, 3) ?? 1; config.PaperWidth = PaperSizeUtils.FromByte((byte)(GetConfig(_mPrinter, RelianceCommands.PaperSizeSub, 0) ?? 80)); + var configrev = GetConfig(_mPrinter, RelianceCommands.GeneralConfigSub, 0x0F); + config.Version = configrev.Version; + config.Revision = configrev.Revision; + } + + /// + /// Reads in and fill config with all bezel parameters + /// + /// Configuration to fill + private void FillBezelConfig(ref RELConfig config) + { + var bezels = new List(); + for (byte i = 0; i < 4; ++i) + { + // [bezel sub] [3] [printer state (i)] + bezels.Add(GetConfig(_mPrinter, RelianceCommands.BezelSub, 3, i)); + } + + config.BezelIdleDutyCycle = bezels[0].DutyCycle; + config.BezelIdleInterval = bezels[0].FlashInterval; + config.BezelPrintingDutyCycle = bezels[1].DutyCycle; + config.BezelPrintingInterval = bezels[1].FlashInterval; + config.BezelPresentedDutyCycle = bezels[2].DutyCycle; + config.BezelPresentedInterval = bezels[2].FlashInterval; + config.BezelEjectingDutyCycle = bezels[3].DutyCycle; + config.BezelEjectingInterval = bezels[3].FlashInterval; + } + + /// + /// Reads in and fills config with all font parameters + /// + /// Configuration to fill + private void FillFontConfig(ref RELConfig config) + { var font = GetConfig(_mPrinter, RelianceCommands.FontSub, 0); config.FontWhich = font.FontWhich; config.FontSize = font.FontSize; config.DefaultCodepage = font.CodePage; - config.FontScalingMode = (RelianceScalarMode) (GetConfig(_mPrinter, RelianceCommands.FontSub, 7) ?? 1); + config.FontScalingMode = (RelianceScalarMode)(GetConfig(_mPrinter, RelianceCommands.FontSub, 7) ?? 1); var codepages = _mPrinter.GetInstalledCodepages().ToList(); config.Codepage1 = codepages.Count > 0 ? codepages[0] : font.CodePage; @@ -185,35 +227,34 @@ public BinaryFile ReadConfiguration() config.IsPaperSlackEnabeld = GetConfig(_mPrinter, RelianceCommands.GeneralConfigSub, 3) == 1; config.IsUniqueUSBSNEnabled = GetConfig(_mPrinter, RelianceCommands.GeneralConfigSub, 7) == 1; - var bezels = new List(); - for (byte i = 0; i < 4; ++i) - { - // [bezel sub] [3] [printer state (i)] - bezels.Add(GetConfig(_mPrinter, RelianceCommands.BezelSub, 3, i)); + } + + /// + /// Reads Serial configuration into config reference + /// + /// Configuration to fill + /// True on succes, else false + private bool FillSerialConfig(ref RELConfig config) + { + var serial = GetConfig(_mPrinter, RelianceCommands.GetSerialConfig); + if (serial == null) + { + // Read failure + return false; } - config.BezelIdleDutyCycle = bezels[0].DutyCycle; - config.BezelIdleInterval = bezels[0].FlashInterval; - config.BezelPrintingDutyCycle = bezels[1].DutyCycle; - config.BezelPrintingInterval = bezels[1].FlashInterval; - config.BezelPresentedDutyCycle = bezels[2].DutyCycle; - config.BezelPresentedInterval = bezels[2].FlashInterval; - config.BezelEjectingDutyCycle = bezels[3].DutyCycle; - config.BezelEjectingInterval = bezels[3].FlashInterval; + config.BaudRate = serial.BaudRate; + config.Handshake = serial.Handshake; + config.Databits = serial.Databits; + config.Stopbits = serial.Stopbits; + config.Parity = serial.Parity; + var xonxoff = GetConfig(_mPrinter, RelianceCommands.GeneralConfigSub, 0x0B); config.XonCode = xonxoff.Xon; config.XoffCode = xonxoff.Xoff; - var configrev = GetConfig(_mPrinter, RelianceCommands.GeneralConfigSub, 0x0F); - config.Version = configrev.Version; - config.Revision = configrev.Revision; - - using (var stream = new MemoryStream()) - { - config.Save(stream); - return BinaryFile.From(stream.GetBuffer()); - } + return true; } /// diff --git a/PTIRelianceLib/IO/HIDPort.cs b/PTIRelianceLib/IO/HIDPort.cs index bae4319..4366e69 100644 --- a/PTIRelianceLib/IO/HIDPort.cs +++ b/PTIRelianceLib/IO/HIDPort.cs @@ -9,7 +9,6 @@ namespace PTIRelianceLib.IO { - using System.Data; using Logging; using Internal; using Transport; diff --git a/PTIRelianceLib/IO/Internal/HIDReport.cs b/PTIRelianceLib/IO/Internal/HIDReport.cs index d902741..0c83a0c 100644 --- a/PTIRelianceLib/IO/Internal/HIDReport.cs +++ b/PTIRelianceLib/IO/Internal/HIDReport.cs @@ -62,7 +62,8 @@ public byte[] GetPayload() { // [id] [length] [... payload ...] var payload = new byte[Data[1]]; - Array.Copy(Data, 2, payload, 0, payload.Length); + var copylen = Math.Min(payload.Length, Data.Length - 2); + Array.Copy(Data, 2, payload, 0, copylen); return payload; } diff --git a/PTIRelianceLib/IO/StructuredReader.cs b/PTIRelianceLib/IO/StructuredReader.cs new file mode 100644 index 0000000..3738eb4 --- /dev/null +++ b/PTIRelianceLib/IO/StructuredReader.cs @@ -0,0 +1,61 @@ +#region Header +// StructuredReader.cs +// PTIRelianceLib +// Cory Todd +// 19-06-2018 +// 8:43 AM +#endregion + +namespace PTIRelianceLib.IO +{ + using System.Collections.Generic; + using Protocol; + using Transport; + + /// + /// Reads pages from an IO device and returns result. + /// + internal class StructuredReader + { + private readonly IPort _device; + + public StructuredReader(IPort device) + { + _device = device; + } + + /// + /// Reads readlen bytes using preamble read command + /// + /// Total bytes to read (not including packet overhead) + /// Command the reads data from data + /// Data that was read + public IPacket Read(int readLen, params byte[] preamble) + { + var buffer = new List(); + var sequenceNum = 0; + + while (buffer.Count < readLen) + { + var cmd = _device.Package(preamble); + var seq = sequenceNum.ToBytesBE(); + cmd.Add(seq); + + _device.Write(cmd); + + var resp = _device.Read(50); + if (resp.GetPacketType() != PacketTypes.PositiveAck) + { + break; + } + + // Extract payload + resp = resp.ExtractPayload(); + buffer.AddRange(resp.GetBytes()); + ++sequenceNum; + } + + return _device.Package(buffer.ToArray()); + } + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Imaging/BasePrintLogo.cs b/PTIRelianceLib/Imaging/BasePrintLogo.cs index 3d8196f..d89a078 100644 --- a/PTIRelianceLib/Imaging/BasePrintLogo.cs +++ b/PTIRelianceLib/Imaging/BasePrintLogo.cs @@ -105,14 +105,8 @@ public void ApplyDithering(DitherAlgorithms ditherAlgorithm, byte threshhold = 1 /// public byte[] ToBuffer() { - if (ImageData == null) - { - return new byte[0]; - } - using (var bitmap = ImageData) - { - return bitmap.ToLogoBuffer(); - } + // Do not wrap in "using" statement as this will will dispose ImageData on closure + return ImageData == null ? new byte[0] : ImageData.ToLogoBuffer(); } /// diff --git a/PTIRelianceLib/Imaging/Dither.cs b/PTIRelianceLib/Imaging/Dither.cs index 3b49939..976d927 100644 --- a/PTIRelianceLib/Imaging/Dither.cs +++ b/PTIRelianceLib/Imaging/Dither.cs @@ -224,15 +224,8 @@ protected virtual void ApplySmoothing( /// byte private static byte SafeByteCast(int val) { - if (val < 0) - { - val = 0; - } - else if (val > 255) - { - val = 255; - } - + val = Math.Max(0, val); + val = Math.Min(255, val); return (byte) val; } } diff --git a/PTIRelianceLib/Imaging/ImageExt.cs b/PTIRelianceLib/Imaging/ImageExt.cs index 0bc5718..880e6c5 100644 --- a/PTIRelianceLib/Imaging/ImageExt.cs +++ b/PTIRelianceLib/Imaging/ImageExt.cs @@ -88,26 +88,8 @@ public static Bitmap AsBitmap(this byte[] imageData, int width, int height) // Timing benchmarks show that safe vs. unsafe has zero impact on our use case // Both safe and unsafe, when tested through logo write, get a 13.2Kbps throughput // This bitmap operation is negligable compared to how slowly the flash data is written. -#if SAFE - // Iterate through rows and columns - for (var row = 0; row < height; row++) - { - // Source is ARGB format but dest is 1bpp so user two different indexers - for (int col = 0, pixCol=0; col < width; col += 4, ++pixCol) - { - var index = row * width + col; - var color = imageData[index++] | - imageData[index++] << 8 | - imageData[index++] << 16 | - imageData[index] << 24; - - // Set pixel - result.SetPixel(pixCol, row, Color.FromArgb(color)); - } - } - -#else - // Lock the entire bitmap +#if UNSAFE +// Lock the entire bitmap var bitmapData = result.LockBits( new Rectangle(0, 0, width, height), ImageLockMode.ReadWrite, @@ -137,6 +119,23 @@ public static Bitmap AsBitmap(this byte[] imageData, int width, int height) // Unlock the bitmap result.UnlockBits(bitmapData); +#else + // Iterate through rows and columns + for (var row = 0; row < height; row++) + { + // Source is ARGB format but dest is 1bpp so user two different indexers + for (int col = 0, pixCol = 0; col < width; col += 4, ++pixCol) + { + var index = row * width + col; + var color = imageData[index++] | + imageData[index++] << 8 | + imageData[index++] << 16 | + imageData[index] << 24; + + // Set pixel + result.SetPixel(pixCol, row, Color.FromArgb(color)); + } + } #endif return result; @@ -176,7 +175,7 @@ public static void InvertColorChannels(this Bitmap bitmapImage) Marshal.Copy(bitmapBgra, 0, bmpWo.Scan0, bmpLen); bitmapImage.UnlockBits(bmpWo); } - + /// /// Creates an MSB ordered, bit-reversed buffer of the logo data located in this bitmap. /// The input data's pixels are reduced into an 8bpp image. That means that 8 PC bitmap diff --git a/PTIRelianceLib/Imaging/LogoSize.cs b/PTIRelianceLib/Imaging/LogoSize.cs index 80e17fd..71ff5af 100644 --- a/PTIRelianceLib/Imaging/LogoSize.cs +++ b/PTIRelianceLib/Imaging/LogoSize.cs @@ -34,14 +34,5 @@ internal class LogoSize /// Gets or sets the size in bytes for this logo /// public int SizeInBytes { get; set; } - - /// - /// Returns dimension string as WidthxHeight - /// - /// - public object GetBitmapSizeString() - { - return string.Format("{0}x{1}", WidthDots, Height); - } } } \ No newline at end of file diff --git a/PTIRelianceLib/Internal/FixedArray.cs b/PTIRelianceLib/Internal/FixedArray.cs new file mode 100644 index 0000000..b92336d --- /dev/null +++ b/PTIRelianceLib/Internal/FixedArray.cs @@ -0,0 +1,80 @@ +#region Header +// FixedArray.cs +// PTIRelianceLib +// Cory Todd +// 19-06-2018 +// 7:50 AM +#endregion + +namespace PTIRelianceLib +{ + using System.Collections.Generic; + using System.Collections.Immutable; + + /// + /// A fixed length array wrapper + /// + /// Data type to hold + public struct FixedArray + { + private readonly IList _mData; + + /// + /// Create a new fixed length array of this type + /// + /// + public FixedArray(int size) + { + Size = size; + _mData = new List(size); + } + + /// + /// Returns the fixed length of this array. This is the max size. + /// + public int Size { get; } + + /// + /// Returns the count of elements in the fixed array. + /// + public int Count => _mData.Count; + + /// + /// Raw raw contents of this array as a comma-separate string using + /// ToString on the objects in this array. + /// + public string Content => string.Join(",", _mData); + + /// + /// Puts data into this fixed length array starting from the current tail + /// of the fixed array. If the fixed array size limit has been reached, + /// no more data will be added and this method will return. + /// + /// + public void SetData(params T[] data) + { + foreach(var d in data) + { + if (_mData.Count >= Size) + { + break; + } + _mData.Add(d); + } + } + + /// + /// Returns a readonly array containg all data + /// + /// + public ImmutableArray GetData() => _mData.ToImmutableArray(); + + /// + /// Returns the element at specified index + /// + /// Index of element to retrieve + /// Returns a reference to object so take care with mutations + /// Object at index + internal T this[int index] => _mData[index]; + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Logo/RELLogoParser.cs b/PTIRelianceLib/Logo/RELLogoParser.cs index 2e1cd0d..84eb1cb 100644 --- a/PTIRelianceLib/Logo/RELLogoParser.cs +++ b/PTIRelianceLib/Logo/RELLogoParser.cs @@ -40,6 +40,7 @@ internal class RELLogoParser /// /// Parse file as a logo /// + /// Starting storage address /// Logo data to parse /// Packetized list or null on error public IList Parse(uint startAddress, byte[] data) diff --git a/PTIRelianceLib/Logo/RELLogoUpdater.cs b/PTIRelianceLib/Logo/RELLogoUpdater.cs index 368803e..b67ff83 100644 --- a/PTIRelianceLib/Logo/RELLogoUpdater.cs +++ b/PTIRelianceLib/Logo/RELLogoUpdater.cs @@ -26,7 +26,7 @@ internal class RELLogoUpdater : IFlashUpdater /// Constructs a new flasher /// /// Port to write to - /// Data to write + /// Data to write /// Thrown if port or fileToFlash are null public RELLogoUpdater(IPort port, RELLogoHeader header) { diff --git a/PTIRelianceLib/PTIRelianceLib.csproj b/PTIRelianceLib/PTIRelianceLib.csproj index 61c9921..f40b129 100644 --- a/PTIRelianceLib/PTIRelianceLib.csproj +++ b/PTIRelianceLib/PTIRelianceLib.csproj @@ -10,33 +10,34 @@ A Reliance Thermal printer API that support firmware updates and configuration. This requires the Reliance printer to be connected via USB. 2018 Pyramid Technologies https://pti.mit-license.org/ - Supports flash updating, confguration read, and configuration write - 0.3.1.0 + Supports flash updating, configuration read, and configuration write + 1.0 https://github.com/PyramidTechnologies/PTI.Reliance.Tools https://github.com/PyramidTechnologies/PTI.Reliance.Tools git dotnet-core thermal-printer hid https://pyramidacceptors.com/favicon.ico - 0.3.1.0 + 1.0.0.0 true false - 0.3.1.0 + 1.0.0.0 true true + false bin\x64\Release\netcoreapp2.0\PTIRelianceLib.xml 0 1701;1702;NU5100 - DEBUG;TRACE;RELEASE;NETCOREAPP;NETCOREAPP2_0;SAFE;RELEASE;NETCOREAPP;NETCOREAPP2_0;RELEASE;NETCOREAPP;NETCOREAPP2_0 + RELEASE;NETCOREAPP;NETCOREAPP2_0;;RELEASE;NETCOREAPP;NETCOREAPP2_1 1701;1702;NU5100 embedded true - TRACE;DEBUG;NETCOREAPP;NETCOREAPP2_0;NETCOREAPP;NETCOREAPP2_0;NETCOREAPP;NETCOREAPP2_0;SAFE;NETCOREAPP;NETCOREAPP2_0;NETCOREAPP;NETCOREAPP2_0 + TRACE;DEBUG;NETCOREAPP;NETCOREAPP2_0;NETCOREAPP;NETCOREAPP2_1 false @@ -62,8 +63,8 @@ - + diff --git a/PTIRelianceLib/Protocol/RelianceCommands.cs b/PTIRelianceLib/Protocol/RelianceCommands.cs index 93b7e33..fe8219a 100644 --- a/PTIRelianceLib/Protocol/RelianceCommands.cs +++ b/PTIRelianceLib/Protocol/RelianceCommands.cs @@ -263,6 +263,11 @@ internal enum RelianceCommands /// AutocutSub = 0x88, + /// + /// Subcommand for telemetry commands + /// + TelemtrySub = 0x90, + /// /// Perform general configuration /// diff --git a/PTIRelianceLib/ReliancePrinter.cs b/PTIRelianceLib/ReliancePrinter.cs index 41150b8..13babf5 100644 --- a/PTIRelianceLib/ReliancePrinter.cs +++ b/PTIRelianceLib/ReliancePrinter.cs @@ -23,6 +23,7 @@ namespace PTIRelianceLib using Logging; using Logo; using Protocol; + using Telemetry; using Transport; /// @@ -77,8 +78,8 @@ public ReliancePrinter() { VendorId = VendorId, ProductId = ProductId, - InReportLength = 34, - OutReportLength = 34, + InReportLength = 35, + OutReportLength = 35, InReportId = 2, OutReportId = 1, NativeHid = new NativeMethods() @@ -233,11 +234,16 @@ public Revlev GetFirmwareRevision() return new Revlev(); } - var cmd = new ReliancePacket(RelianceCommands.GetRevlev); - // "Self" param specifies we want revlev for running application - cmd.Add(0x10); + Log.Debug("Requesting revision level"); + + // 0x10: specifies we want revlev for running application + var cmd = _mPort.Package((byte)RelianceCommands.GetRevlev, 0x10); + var resp = Write(cmd); - return PacketParserFactory.Instance.Create().Parse(resp); + var rev = PacketParserFactory.Instance.Create().Parse(resp); + + Log.Debug("Found firmware revision {0}", rev); + return rev; } /// @@ -287,6 +293,8 @@ public ReturnCodes Ping() return ReturnCodes.DeviceNotConnected; } + Log.Debug("Sending Ping"); + var cmd = new ReliancePacket(RelianceCommands.Ping); var resp = Write(cmd); return resp.GetPacketType() == PacketTypes.PositiveAck ? ReturnCodes.Okay : ReturnCodes.ExecutionFailure; @@ -312,6 +320,7 @@ public ReturnCodes Reboot() try { + Log.Debug("Rebooting printer"); var cmd = new ReliancePacket(RelianceCommands.Reboot); Write(cmd); @@ -416,7 +425,8 @@ public IEnumerable GetInstalledCodepages() /// You probably want to configure this yourself. /// Return Code /// Thrown if logodata is null - public ReturnCodes StoreLogos(IList logoData, IProgressMonitor monitor, LogoStorageConfig storageConfig) + public ReturnCodes StoreLogos(IList logoData, IProgressMonitor monitor, + LogoStorageConfig storageConfig) { if (logoData == null) { @@ -436,12 +446,12 @@ public ReturnCodes StoreLogos(IList logoData, IProgressMonitor monit } var logoBank = new RELLogoBank(); - var ditheredLogos = logoData.Select(logo => + var ditheredLogos = logoData.Select(logo => new BasePrintLogo(logo, maxWidth: storageConfig.MaxPixelWidth)).Cast().ToList(); ditheredLogos.ForEach(x => x.ApplyDithering(storageConfig.Algorithm, storageConfig.Threshold)); - + // MakeHeaders locks in our scaling and dithering options - foreach(var header in logoBank.MakeHeaders(ditheredLogos)) + foreach (var header in logoBank.MakeHeaders(ditheredLogos)) { // sub command 2: Set Logo header var cmd = _mPort.Package((byte) RelianceCommands.LogoSub, 0x02); @@ -495,11 +505,117 @@ public ReturnCodes PrintLogo(int index) { return ReturnCodes.InvalidRequestPayload; } + // 7 == print logo sub command Write(RelianceCommands.LogoSub, 7, (byte) index); return ReturnCodes.Okay; } + /// + /// Returns the telemetry data over the lifetime of this printer + /// + /// Requires firmware 1.28+. Older firmware will result in null result. + /// LifetimeTelemtry data since printer left factory + public LifetimeTelemetry GetLifetimeTelemetry() + { + if (GetFirmwareRevision() < new Revlev("1.28")) + { + // unsupported + return null; + } + + // 1: read non-volatile version, from start to end + return (LifetimeTelemetry) ReadTelemetry(TelemetryTypes.Lifetime); + } + + /// + /// Returns the telemetry data of this printer since last power up + /// + /// Requires firmware 1.28+. Older firmware will result in null result. + /// Powerup telemetry or null if read failure or unsupported firmware + public PowerupTelemetry GetPowerupTelemetry() + { + // 2: read volatile version, from start to end + return GetFirmwareRevision() < new Revlev("1.28") ? null : ReadTelemetry(TelemetryTypes.Powerup); + } + + /// + /// Reset all telemetry counters. This include the lifetime and powerup counters. + /// This operation cannot be undone. + /// + /// Data will be permanently erased + /// Success code + public ReturnCodes ResetTelemetry() + { + var resp = Write(RelianceCommands.TelemtrySub, 0x03); + return resp.GetPacketType() == PacketTypes.PositiveAck ? ReturnCodes.Okay : ReturnCodes.ExecutionFailure; + } + + /// + /// Reads the specified telemtry data block + /// + /// Type of telemetry to request + /// + internal PowerupTelemetry ReadTelemetry(TelemetryTypes type) + { + Log.Debug("Requesting telemetry info"); + + var readlen = ReadTelemetrySize(type); + if (readlen <= 0) + { + // Bad read size + return null; + } + + // Build permission request. 0: request data + var request = _mPort.Package((byte) RelianceCommands.TelemtrySub, 0, (byte) type); + + // Read entire data chunk from start of data (0->readlen) + request.Add(((ushort) 0).ToBytesBE()); + request.Add(((ushort) readlen).ToBytesBE()); + + // Request permission + var resp = Write(request); + if (resp.GetPacketType() != PacketTypes.PositiveAck) + { + // Permission denied + return null; + } + + // 1: repeatedly read data request + var preamble = new byte[] {(byte) RelianceCommands.TelemtrySub, 1}; + + // Execute a structured read + var reader = new StructuredReader(_mPort); + resp = reader.Read(readlen, preamble); + return PacketParserFactory.Instance.Create().Parse(resp); + } + + /// + /// Reads the size of the specified telemetry struct from target + /// + /// Telemetry format to request sizeof + /// Integer size in bytes, -1 on error + internal int ReadTelemetrySize(TelemetryTypes type) + { + // 4: read data size request + var getSize = _mPort.Package((byte) RelianceCommands.TelemtrySub, 4, (byte) type); + var resp = Write(getSize); + if (resp.GetPacketType() != PacketTypes.PositiveAck) + { + return -1; + } + + var readlen = PacketParserFactory.Instance.Create().Parse(resp).Value; + if (readlen <= 0) + { + // Invalid data read length + return -1; + } + + return readlen; + } + /// /// Returns firmware application id string or empty on error /// @@ -511,6 +627,7 @@ internal string GetAppId() return string.Empty; } + Log.Debug("Requesting app id"); var cmd = new ReliancePacket((byte) RelianceCommands.GetBootId, 0x10); var resp = Write(cmd); return PacketParserFactory.Instance.Create().Parse(resp).Value; @@ -527,6 +644,7 @@ internal ReturnCodes EnterBootloader() return ReturnCodes.DeviceNotConnected; } + Log.Debug("Entering bootloader"); var resp = Write(RelianceCommands.SetBootMode, 0x21); return resp.GetPacketType() != PacketTypes.PositiveAck ? ReturnCodes.FailedBootloaderEntry : Reboot(); diff --git a/PTIRelianceLib/Status.cs b/PTIRelianceLib/Status.cs index 2be9ce0..8a4e7f0 100644 --- a/PTIRelianceLib/Status.cs +++ b/PTIRelianceLib/Status.cs @@ -30,7 +30,7 @@ public class Status : IParseable /// /// 12-bit ADC factor for a 3.3 volt reference /// - private const float _mAdcConstant = 0.000806f; + private const float AdcConstant = 0.000806f; /// /// ASCII string of the head input voltage. "XX.XX" (Volts) @@ -108,11 +108,11 @@ public override string ToString() sb.AppendFormat("Head Temperature: {0} °C\n", HeadTemp); sb.AppendFormat("Head Temperature: {0} °F\n", (HeadTemp * (9 / 5.0) + 32).ToString("N0")); sb.AppendFormat("Sensor Status: {0} \n", SensorStatus); - sb.AppendFormat("Presenter Raw: {0} V DC\n", PresenterRaw * _mAdcConstant); - sb.AppendFormat("Path Raw: {0} V DC\n", PathRaw * _mAdcConstant); - sb.AppendFormat("Paper Raw: {0} V DC\n", PaperRaw * _mAdcConstant); - sb.AppendFormat("Notch Raw: {0} V DC\n", NotchRaw * _mAdcConstant); - sb.AppendFormat("Arm Raw: {0} V DC\n", ArmRaw * _mAdcConstant); + sb.AppendFormat("Presenter Raw: {0} V DC\n", PresenterRaw * AdcConstant); + sb.AppendFormat("Path Raw: {0} V DC\n", PathRaw * AdcConstant); + sb.AppendFormat("Paper Raw: {0} V DC\n", PaperRaw * AdcConstant); + sb.AppendFormat("Notch Raw: {0} V DC\n", NotchRaw * AdcConstant); + sb.AppendFormat("Arm Raw: {0} V DC\n", ArmRaw * AdcConstant); sb.AppendFormat("Ticket Status: {0} \n", TicketStatus); sb.AppendFormat("Errors Status: {0} \n", PrinterErrors); diff --git a/PTIRelianceLib/Telemetry/EjectionStatus.cs b/PTIRelianceLib/Telemetry/EjectionStatus.cs new file mode 100644 index 0000000..5275a47 --- /dev/null +++ b/PTIRelianceLib/Telemetry/EjectionStatus.cs @@ -0,0 +1,30 @@ +#region Header +// EjectionStatus.cs +// PTIRelianceLib +// Cory Todd +// 26-06-2018 +// 11:34 AM +#endregion + +namespace PTIRelianceLib.Telemetry +{ + /// + /// A ticket, once printed, terminates in one of two states. On + /// powerup, the floating state is called NoTicket. + /// + public enum EjectionStatus + { + /// + /// No ticket state on record + /// + NoTicket = 0, + /// + /// Last ticket was ejected from bezel + /// + Ejected = 1, + /// + /// Last ticket was retracted + /// + Retracted = 2 + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Telemetry/LastTicketState.cs b/PTIRelianceLib/Telemetry/LastTicketState.cs new file mode 100644 index 0000000..edbea93 --- /dev/null +++ b/PTIRelianceLib/Telemetry/LastTicketState.cs @@ -0,0 +1,31 @@ +#region Header +// LastTicketState.cs +// PTIRelianceLib +// Cory Todd +// 26-06-2018 +// 11:34 AM +#endregion + +namespace PTIRelianceLib.Telemetry +{ + /// + /// Each ticket print is tracked by ejection and length + /// + public struct LastTicketState + { + /// + /// How was the last ticket handled + /// + public EjectionStatus Status { get; set; } + + /// + /// Length of the last ticket in mm + /// + public int LengthMm { get; set; } + + /// + /// Reserved + /// + internal int Reserved { get; set; } + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Telemetry/LifetimeTelemetry.cs b/PTIRelianceLib/Telemetry/LifetimeTelemetry.cs new file mode 100644 index 0000000..386114b --- /dev/null +++ b/PTIRelianceLib/Telemetry/LifetimeTelemetry.cs @@ -0,0 +1,137 @@ +#region Header +// LifetimeTelemetry.cs +// PTIRelianceLib +// Cory Todd +// 19-06-2018 +// 7:42 AM +#endregion + +namespace PTIRelianceLib.Telemetry +{ + using System.IO; + using Transport; + + /// + /// + /// Telemetry data for the lifetime of the printer. + /// This data persists across flash updates and configuration changes. + /// + public class LifetimeTelemetry : PowerupTelemetry + { + /// + /// How many times the printer has powered up. + /// + public int PowerUpCount { get; set; } + + /// + /// How many times has the HID reset command been received. + /// + public int ResetCmdCount { get; set; } + + /// + public override byte[] Serialize() + { + // TODO implement telemetry serializer + return new byte[0]; + } + } + + /// + /// + /// Parser class for consumes an + /// + /// and produces a . + /// + internal class LifetimeTelemetryParser : BaseModelParser + { + public override LifetimeTelemetry Parse(IPacket packet) + { + // Needs a valid packet and at least 8 bytes to identify type + packet = CheckPacket(packet); + if (packet == null || packet.Count < 8) + { + return null; + } + + var tel = new LifetimeTelemetry(); + using (var stream = new MemoryStream(packet.GetBytes())) + using(var reader = new BinaryReader(stream)) + { + tel.StructRevision = reader.ReadInt32(); + tel.StructSize = reader.ReadInt32(); + + // is the size logical? + if (tel.StructSize == 0 || tel.StructSize == -1) + { + return null; + } + + // Ensure there is enough data left in the stream to read + if (packet.Count < tel.StructSize) + { + return null; + } + + tel.PowerUpCount = reader.ReadInt32(); + tel.CutterCount = reader.ReadInt32(); + tel.PlatenOpenCount = reader.ReadInt32(); + tel.PaperMovedCount = reader.ReadInt32(); + tel.TicketCount = reader.ReadInt32(); + tel.TicketsRetracted = reader.ReadInt32(); + tel.TicketsEjected = reader.ReadInt32(); + tel.TicketsPulled = reader.ReadInt32(); + + // Read in and parse raw data in appropriate width + var rawData = reader.ReadBytes(PowerupTelemetry.LengthLogLength * 4); + using (var blockStream = new MemoryStream(rawData)) + using (var blockReader = new BinaryReader(blockStream)) + { + // Read in presented length log, contains 4 byte integers + tel.TicketLengthLog = new FixedArray(PowerupTelemetry.LengthLogLength); + for (var i = 0; i < PowerupTelemetry.LengthLogLength; ++i) + { + tel.TicketLengthLog.SetData(blockReader.ReadInt32()); + } + } + + + // Read in and parse raw data in appropriate width + rawData = reader.ReadBytes(PowerupTelemetry.PresentLogLength * 4); + using (var blockStream = new MemoryStream(rawData)) + using (var blockReader = new BinaryReader(blockStream)) + { + // Read in presented length log, contains 4 byte integers + tel.TicketPullTimeLog = new FixedArray(PowerupTelemetry.PresentLogLength); + for (var i = 0; i < PowerupTelemetry.PresentLogLength; ++i) + { + tel.TicketPullTimeLog.SetData(blockReader.ReadInt32()); + } + } + + tel.AvgTimePresented = reader.ReadInt32(); + tel.Button = reader.ReadInt16(); + tel.PaperOutDuringPrint = reader.ReadInt16(); + tel.PaperOut = reader.ReadInt16(); + tel.AvgCutTime = reader.ReadInt16(); + tel.ResetCmdCount = reader.ReadInt32(); + tel.JammedCount = reader.ReadInt16(); + tel.OverheatedCount = reader.ReadInt32(); + tel.CriticalErrorCount = reader.ReadInt32(); + tel.HighErrorCount = reader.ReadInt16(); + + // Start rev3 attributes + if (tel.StructRevision >= 3) + { + tel.LastTicketState = new LastTicketState + { + Status = (EjectionStatus) reader.ReadInt32(), + LengthMm = reader.ReadInt32(), + Reserved = reader.ReadInt32(), + }; + } + } + + return tel; + } + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Telemetry/PowerupTelemetry.cs b/PTIRelianceLib/Telemetry/PowerupTelemetry.cs new file mode 100644 index 0000000..f5c4394 --- /dev/null +++ b/PTIRelianceLib/Telemetry/PowerupTelemetry.cs @@ -0,0 +1,194 @@ +#region Header + +// RELTelemetry.cs +// PTIRelianceLib +// Cory Todd +// 19-06-2018 +// 7:19 AM + +#endregion + +namespace PTIRelianceLib.Telemetry +{ + using Transport; + + /// + /// + /// Contains telemtry data describing the usage and lifetime of a Reliance + /// thermal printer. The data is produced by a Reliance printer and + /// handled by this API. + /// + public class PowerupTelemetry : IParseable + { + internal const int LengthLogLength = 9; + internal const int PresentLogLength = 11; + + /// + /// The revision of this data structure. + /// + internal int StructRevision { get; set; } + + /// + /// Size in bytes of this data structure. + /// + internal int StructSize { get; set; } + + /// + /// How many cut cycles has there been. + /// + public int CutterCount { get; set; } + + /// + /// How many times has the head been opened. + /// + public int PlatenOpenCount { get; set; } + + /// + /// How much paper has been moved by the printer + /// in it's lifetime in units of steps. 1 step = 0.125mm + /// + public int PaperMovedCount { get; set; } + + /// + /// How many tickets have been printed. Regardless of size. + /// + public int TicketCount { get; set; } + + /// + /// How many tickets have been retracted. + /// + public int TicketsRetracted { get; set; } + + /// + /// How many tickets have been ejected. + /// + public int TicketsEjected { get; set; } + + /// + /// How many tickets have been pulled by the customer. + /// + public int TicketsPulled { get; set; } + + /// + /// Each index is a ticket length range. There are 9 + /// elements in this array. + /// +#if DEBUG + public +#else + internal +#endif + FixedArray TicketLengthLog { get; set; } + + /// + /// Each index is the time a pulled ticket was sitting + /// at the bezel. Only pulled tickets are logged. There + /// are 11 elements in this array. + /// +#if DEBUG + public +#else + internal +#endif + FixedArray TicketPullTimeLog { get; set; } + + /// + /// Avg time pulled tickets are sitting at the bezel. + /// + public int AvgTimePresented { get; set; } + + /// + /// How many times the push button action was used. + /// + public short Button { get; set; } + + /// + /// How many times printer ran out of paper during a print job. + /// + public short PaperOutDuringPrint { get; set; } + + /// + /// How many times has the printer ran out of paper including during a print job. + /// + public short PaperOut { get; set; } + + /// + /// Keeps track of the avg cut time for up to the last 500 cuts. + /// + public short AvgCutTime { get; set; } + + /// + /// How many times the printer has jammed. + /// + public short JammedCount { get; set; } + + /// + /// How many times the printer has over heated. + /// + public int OverheatedCount { get; set; } + + /// + /// How many times the printer was in a critical error. + /// + public int CriticalErrorCount { get; set; } + + /// + /// How many times the printer was in a high priority error. + /// + public short HighErrorCount { get; set; } + + /// + /// State of the last ticket + /// + public LastTicketState LastTicketState { get; set; } + + /// + /// Returns the count of tickets within the specified ticket + /// + /// Length group to retrieve + /// Number of tickets printed in specified length group + public int TicketCountsByLength(TicketLengthGroups group) => TicketLengthLog[(int) group]; + + /// + /// Returns the count of tickets that pulled by a customer within the specified time bin. + /// These counts only include tickets pull by the customer. That means that retractions and + /// ejections are not included in this metric. + /// + /// Time group to retrieve + /// Number of tickets in specified time group + public int TicketCountByTimeToPull(TicketPullTimeGroups group) => TicketPullTimeLog[(int) group]; + + /// + public virtual byte[] Serialize() + { + // TODO implement PowerupTelemetry serializer + return new byte[0]; + } + } + + /// + /// + /// Parser class for consumes an + /// + /// and produces a . + /// + internal class PowerupTelemetryParser : BaseModelParser + { + public override PowerupTelemetry Parse(IPacket packet) + { + var parser = new LifetimeTelemetryParser(); + var tel = parser.Parse(packet); + if (tel == null) + { + // Parse failed + return null; + } + + // These two fields don't make sense for power up context + tel.PowerUpCount = 0; + tel.ResetCmdCount = 0; + + return tel; + } + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Telemetry/TelemetryTypes.cs b/PTIRelianceLib/Telemetry/TelemetryTypes.cs new file mode 100644 index 0000000..703c341 --- /dev/null +++ b/PTIRelianceLib/Telemetry/TelemetryTypes.cs @@ -0,0 +1,25 @@ +#region Header +// TelemetryTypes.cs +// PTIRelianceLib +// Cory Todd +// 25-06-2018 +// 7:40 AM +#endregion + +namespace PTIRelianceLib.Telemetry +{ + /// + /// Context of telemtry data + /// + internal enum TelemetryTypes + { + /// + /// Data is counted since factory manugacturing + /// + Lifetime = 0, + /// + /// Data is counted since last powerup + /// + Powerup = 1, + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Telemetry/TicketLengthGroups.cs b/PTIRelianceLib/Telemetry/TicketLengthGroups.cs new file mode 100644 index 0000000..a7595b8 --- /dev/null +++ b/PTIRelianceLib/Telemetry/TicketLengthGroups.cs @@ -0,0 +1,55 @@ +#region Header +// TicketLengthGroups.cs +// PTIRelianceLib +// Cory Todd +// 25-06-2018 +// 10:53 AM +#endregion + +namespace PTIRelianceLib.Telemetry +{ + /// + /// Ticket length tracking uses binning to group count of ticket lengths + /// together. There are 9 length groups that are tracked by length + /// in millimeters. + /// + public enum TicketLengthGroups + { + /// + /// Ticket lengths up to 86.375 mm + /// + Bin86 = 0, + /// + /// Ticket lengths from to 118.125 mm + /// + Bin118, + /// + /// Ticket lengths from 118.125 mm to 149.875 mm + /// + Bin149, + /// + /// Ticket lengths from 149.875 mm to 187.875 mm + /// + Bin187, + /// + /// Ticket lengths from 187.875 mm to 226 mm + /// + Bin226, + /// + /// Ticket lengths from 226 mm to 276.875 mm + /// + Bin276, + /// + /// Ticket lengths from 276.875 mm to 327.625 mm + /// + Bin327, + /// + /// Ticket lengths from 327.625 mm to 442 mm + /// + Bin442, + /// + /// Tickets lengths exceeding 442 mm + /// + BinOversized, + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Telemetry/TicketPullTimeGroups.cs b/PTIRelianceLib/Telemetry/TicketPullTimeGroups.cs new file mode 100644 index 0000000..621a8fe --- /dev/null +++ b/PTIRelianceLib/Telemetry/TicketPullTimeGroups.cs @@ -0,0 +1,64 @@ +#region Header +// TicketPullTimeGroups.cs +// PTIRelianceLib +// Cory Todd +// 25-06-2018 +// 11:29 AM +#endregion + +namespace PTIRelianceLib.Telemetry +{ + /// + /// Each printed ticket that sits at the bezel starts a clock. The time between + /// print presented and a customer pulling the ticket is is tracked using binning. + /// The times are tracked in whole seconds. Only tickets that are pulled are counted + /// in these groupings. i.e. Ticket retraction and auto-ejected tickets are not counted. + /// + public enum TicketPullTimeGroups + { + /// + /// Ticket pulled within 4 seconds + /// + Bin4, + /// + /// Ticket pull took more than 4 seconds, up to 8 seconds + /// + Bin8, + /// + /// Ticket pull took more than 8 seconds, up to 16 seconds + /// + Bin16, + /// + /// Ticket pull took more than 16 seconds, up to 32 seconds + /// + Bin32, + /// + /// Ticket pull took more than 32 seconds, up to 50 seconds + /// + Bin50, + /// + /// Ticket pull took more than 50 seconds, up to 70 seconds + /// + Bin70, + /// + /// Ticket pull took more than 70 seconds, up to 90 seconds + /// + Bin90, + /// + /// Ticket pull took more than 90 seconds, up to 120 seconds + /// + Bin120, + /// + /// Ticket pull took more than 120 seconds, up to 180 seconds + /// + Bin180, + /// + /// Ticket pull took more than 180 seconds, up to 300 seconds + /// + Bin300, + /// + /// Ticket pull took more than 300 seconds + /// + BinOvertime + } +} \ No newline at end of file diff --git a/PTIRelianceLib/Transport/PacketParserFactory.cs b/PTIRelianceLib/Transport/PacketParserFactory.cs index 94e568b..7685536 100644 --- a/PTIRelianceLib/Transport/PacketParserFactory.cs +++ b/PTIRelianceLib/Transport/PacketParserFactory.cs @@ -11,6 +11,7 @@ namespace PTIRelianceLib.Transport using System; using System.Collections.Generic; using Configuration; + using Telemetry; /// /// Typesafe parser locator factory. These functions extended objects known to implement IParseAs. @@ -32,6 +33,8 @@ static PacketParserFactory() Instance.Register(); Instance.Register(); Instance.Register(); + Instance.Register(); + Instance.Register(); } private PacketParserFactory() { } @@ -48,6 +51,7 @@ private PacketParserFactory() { } /// /// Class to generate /// Parser class + /// Thrown if T or K are not instantiable public void Register() where T : IParseable where TK : IParseAs @@ -73,6 +77,8 @@ public void Register() /// /// /// + /// Raised if a parser for type T has + /// not been registered. public IParseAs Create(params object[] parameters) where T : IParseable { diff --git a/RelianceCLI/Program.cs b/RelianceCLI/Program.cs index 5c87828..a362d24 100644 --- a/RelianceCLI/Program.cs +++ b/RelianceCLI/Program.cs @@ -5,6 +5,8 @@ namespace RelianceCLI { using System.IO; + using Newtonsoft.Json; + using Newtonsoft.Json.Converters; using PTIRelianceLib.Imaging; using PTIRelianceLib.Protocol; @@ -144,11 +146,25 @@ private static void Run(Options opts) } } + if (opts.LifetimeTelemetry) + { + var resp = printer.GetLifetimeTelemetry(); + var str = JsonConvert.SerializeObject(resp, Formatting.Indented, new StringEnumConverter()); + Console.WriteLine("Lifetime Telementry:\n{0}", str); + } + + if (opts.StartupTelemetry) + { + var resp = printer.GetPowerupTelemetry(); + var str = JsonConvert.SerializeObject(resp, Formatting.Indented, new StringEnumConverter()); + Console.WriteLine("Powerup Telementry:\n{0}", str); + } + if (opts.GetStatus) { var status = printer.GetStatus(); Console.WriteLine("Printer status:\n{0}", status); - Console.WriteLine("Has Paper? :{0}", status.SensorStatus.HasFlag(SensorStatuses.Path)); + Console.WriteLine("Has Paper? :{0}", status.SensorStatus.HasFlag(SensorStatuses.Paper)); } } @@ -174,11 +190,16 @@ private struct Options public string SaveConfigPath; + public bool LifetimeTelemetry; + + public bool StartupTelemetry; + public static Options Parse(IEnumerable args) { var opts = new Options(); Action nextCapture = null; + var lastSwitch = string.Empty; foreach (var str in args) { @@ -197,11 +218,13 @@ public static Options Parse(IEnumerable args) break; case "-f": case "--firmware": + lastSwitch = str; nextCapture = s => opts.FirmwareFilePath = s; break; case "-c": case "--config": + lastSwitch = str; nextCapture = s => opts.ConfigFilePath = s; break; @@ -212,6 +235,7 @@ public static Options Parse(IEnumerable args) case "-l": case "--logo": + lastSwitch = str; nextCapture = s => opts.LogoFilePath = s; break; @@ -222,9 +246,19 @@ public static Options Parse(IEnumerable args) case "-g": case "--get-config": + lastSwitch = str; nextCapture = s => opts.SaveConfigPath = s; break; + case "-t": + case "--startup-telem": + opts.StartupTelemetry = true; + break; + case "-T": + case "--lifetime-telem": + opts.LifetimeTelemetry = true; + break; + default: opts.Message = string.Format("Unknown switch: {0}", str); return opts; @@ -234,7 +268,7 @@ public static Options Parse(IEnumerable args) if (nextCapture != null) { - opts.Message = "Incomplete command line switch"; + opts.Message = string.Format("Incomplete command line switch: {0}", lastSwitch); } else { @@ -257,6 +291,8 @@ public static string Usage() "FLAGS\n" + "\t-r,--revision\t\tRead and display printer firmware revision\n" + "\t-s,--status\t\tRead and display printer status\n" + + "\t-t,--startup-telem\t\tRead startup telemetry\n" + + "\t-T,--lifetime-telem\t\tRead lifetime telemetry\n" + "\t-p,--power\t\tReboot printer immediately\n"; } } diff --git a/RelianceCLI/RelianceCLI.csproj b/RelianceCLI/RelianceCLI.csproj index 6447319..9ceae57 100644 --- a/RelianceCLI/RelianceCLI.csproj +++ b/RelianceCLI/RelianceCLI.csproj @@ -6,6 +6,7 @@ AnyCPU;x64 false false + false diff --git a/RelianceCLI/publish_debug.bat b/RelianceCLI/publish_debug.bat new file mode 100644 index 0000000..4a0144c --- /dev/null +++ b/RelianceCLI/publish_debug.bat @@ -0,0 +1 @@ + dotnet publish -c Release -o build --self-contained -r win10-x64 /p:Platform=x64 \ No newline at end of file diff --git a/docsource/tutorials/Sample_05.cs b/docsource/tutorials/Sample_05.cs index cd20d11..2ff74b1 100644 --- a/docsource/tutorials/Sample_05.cs +++ b/docsource/tutorials/Sample_05.cs @@ -1,4 +1,5 @@ using System; +using System.Collections.Generic; using System.Threading; using PTIRelianceLib; using PTIRelianceLib.Imaging; @@ -9,21 +10,19 @@ class Program { static void Main(string[] args) { - - // Wrap our printer in using so it gets disposed on properly + // Wrap our printer in using so it gets disposed on properly using (var printer = new ReliancePrinter()) { var logos = new List { - BinaryFile.From("my_logo_a.jpg"), - BinaryFile.From("my_logo_b.png"), - BinaryFile.From("another.bmp"), + BinaryFile.From("index.jpg"), + BinaryFile.From("index2.jpg"), }; var config = LogoStorageConfig.Default; config.Algorithm = DitherAlgorithms.Atkinson; - var result = printer.StoreLogos(logos, new ConsoleProgressBar(), config); + var result = printer.StoreLogos(logos, new DevNullMonitor(), config); Console.WriteLine("Write Result: {0}", result); // use logo index to recall and print @@ -34,8 +33,7 @@ static void Main(string[] args) // Give ~3 seconds to print Thread.Sleep(3 * 1000); } - } } } -} +} \ No newline at end of file diff --git a/docsource/tutorials/Sample_06.cs b/docsource/tutorials/Sample_06.cs new file mode 100644 index 0000000..416044e --- /dev/null +++ b/docsource/tutorials/Sample_06.cs @@ -0,0 +1,27 @@ +using System; +using System.Collections.Generic; +using System.Threading; +using PTIRelianceLib; +using Newtonsoft.Json; +using Newtonsoft.Json.Converters; + +namespace reliance_sample +{ + class Program + { + static void Main(string[] args) + { + // Wrap our printer in using so it gets disposed on properly + using (var printer = new ReliancePrinter()) + { + var tel1 = printer.GetPowerupTelemetry(); + var str = JsonConvert.SerializeObject(tel1, Formatting.Indented, new StringEnumConverter()); + Console.WriteLine("Powerup Telementry:\n{0}", str); + + var tel2 = printer.GetLifetimeTelemetry(); + var str = JsonConvert.SerializeObject(tel2, Formatting.Indented, new StringEnumConverter()); + Console.WriteLine("Lifetime Telementry:\n{0}", str); + } + } + } +} \ No newline at end of file diff --git a/docsource/tutorials/telemetry.md b/docsource/tutorials/telemetry.md new file mode 100644 index 0000000..8de3196 --- /dev/null +++ b/docsource/tutorials/telemetry.md @@ -0,0 +1,27 @@ +# Telemetry +The Reliance Thermal printer tracks the metrics that matter. We have counters for just about any even you can imagine and they are all accessible through this API. With this information, you can track paper consumption, ticket pull habits, error patterns, and many other metrics. + +There are two groups for telemetry: + +1. Lifetime : This is a cumulative record of all events since the printer left our factory. +2. Powerup : This is a record of all events since the last power cycle. + +See and for more details. + +> [!Warning] +> Require Firmware 1.28+. Calling this API on older firmware will return null. + +## Ticket Pull +The best way to detect a ticket pull is request the telemetry information and inspect the property. This records the most recent action taken on a printed ticket along with the ticket's length in millimeters. We recommend that you do not poll the printer more than 4 times a second in order to prevent unecessary blocking read requests. After a ticket is printed, poll the printer about once a second and watch for the TicketCount property to increment. Once it increments, you can read the LastTicketState property and examine how the ticket was handled. + + +## Ticket Lengths +Tickets lengths are tracked by binning ticket lengths into 9 groups. The exact lengths groups are enumerated in in millimeters. The telemetry object tracks the count of each ticket by these groups. All tickets with the exception of startup and push-button diagnostic tickets are counted in this metric. + +## Ticket Pull Time +Tickets pull time is the time in second it took for a customer to pull the ticket from the printer. If a ticket is never pulled and is instead ejected or retracted, no measurement will be taken. The exact time groups are enumerated in in second. + +## Code Sample +[!code-csharp[Main](Sample_06.cs)] + +[!include[]()] \ No newline at end of file diff --git a/docsource/tutorials/toc.yml b/docsource/tutorials/toc.yml index c1fe4aa..50c2033 100644 --- a/docsource/tutorials/toc.yml +++ b/docsource/tutorials/toc.yml @@ -8,6 +8,8 @@ href: logos.md - name: Status Check href: status_check.md +- name: Telemetry + href: telemetry.md - name: Library Logging href: logging.md - name: Library Customization