preliminary SLOOOW plugin for reading blockchain databases

This commit is contained in:
Michael Becker 2020-09-20 01:13:11 -04:00
parent 9ac1ea3731
commit 1375a418f9
No known key found for this signature in database
GPG Key ID: 506F54899E2BFED7
15 changed files with 613 additions and 0 deletions

View File

@ -0,0 +1,108 @@
<?xml version="1.0" encoding="UTF-8"?>
<!-- Generated with glade 3.22.2 -->
<interface>
<requires lib="gtk+" version="3.20"/>
<object class="GtkTreeStore" id="tm">
<columns>
<!-- column-name colVersion -->
<column type="gchararray"/>
<!-- column-name colPreviousBlockHash -->
<column type="gchararray"/>
<!-- column-name colMerkelRoot -->
<column type="gchararray"/>
<!-- column-name colTimestamp -->
<column type="gchararray"/>
</columns>
</object>
<object class="GtkWindow">
<property name="can_focus">False</property>
<child type="titlebar">
<placeholder/>
</child>
<child>
<object class="GtkBox">
<property name="visible">True</property>
<property name="can_focus">False</property>
<property name="orientation">vertical</property>
<child>
<object class="GtkScrolledWindow">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="shadow_type">in</property>
<child>
<object class="GtkTreeView" id="tv">
<property name="visible">True</property>
<property name="can_focus">True</property>
<property name="model">tm</property>
<child internal-child="selection">
<object class="GtkTreeSelection"/>
</child>
<child>
<object class="GtkTreeViewColumn" id="tvcVersion">
<property name="resizable">True</property>
<property name="title" translatable="yes">Version</property>
<property name="clickable">True</property>
<property name="reorderable">True</property>
<child>
<object class="GtkCellRendererText"/>
<attributes>
<attribute name="text">0</attribute>
</attributes>
</child>
</object>
</child>
<child>
<object class="GtkTreeViewColumn" id="tvcPreviousBlockHash">
<property name="resizable">True</property>
<property name="title" translatable="yes">Previous Block Hash</property>
<property name="clickable">True</property>
<property name="reorderable">True</property>
<child>
<object class="GtkCellRendererText"/>
<attributes>
<attribute name="text">1</attribute>
</attributes>
</child>
</object>
</child>
<child>
<object class="GtkTreeViewColumn" id="tvcMerkelRoot">
<property name="resizable">True</property>
<property name="title" translatable="yes">Merkel Root</property>
<property name="clickable">True</property>
<property name="reorderable">True</property>
<child>
<object class="GtkCellRendererText"/>
<attributes>
<attribute name="text">2</attribute>
</attributes>
</child>
</object>
</child>
<child>
<object class="GtkTreeViewColumn" id="tvcTimestamp">
<property name="resizable">True</property>
<property name="title" translatable="yes">Timestamp</property>
<property name="clickable">True</property>
<property name="reorderable">True</property>
<child>
<object class="GtkCellRendererText"/>
<attributes>
<attribute name="text">3</attribute>
</attributes>
</child>
</object>
</child>
</object>
</child>
</object>
<packing>
<property name="expand">True</property>
<property name="fill">True</property>
<property name="position">0</property>
</packing>
</child>
</object>
</child>
</object>
</interface>

View File

@ -308,6 +308,7 @@
<Content Include="Editors\RavenSoftware\Icarus\Dialogs\IcarusExpressionHelperDialog.glade" />
<Content Include="Editors\RavenSoftware\Carcass\CarcassEditor.glade" />
<Content Include="Editors\RavenSoftware\Carcass\Dialogs\ModelReferencePropertiesDialog.glade" />
<Content Include="Editors\Blockchain\BlockchainEditor.glade" />
</ItemGroup>
<ItemGroup>
<Content Include="Configuration\Application.upl" />

View File

@ -0,0 +1,81 @@
//
// MyClass.cs
//
// Author:
// Michael Becker <alcexhim@gmail.com>
//
// Copyright (c) 2020 Mike Becker's Software
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
using System;
using MBS.Framework.UserInterface;
using MBS.Framework.UserInterface.Controls.ListView;
using UniversalEditor.Plugins.Blockchain.Bitcoin.ObjectModels;
using UniversalEditor.UserInterface;
namespace UniversalEditor.Plugins.Blockchain.UserInterface
{
[ContainerLayout("~/Editors/Blockchain/BlockchainEditor.glade")]
public class BlockchainEditor : Editor
{
private ListViewControl tv;
private static EditorReference _er = null;
public override EditorReference MakeReference()
{
if (_er == null)
{
_er = base.MakeReference();
_er.SupportedObjectModels.Add(typeof(BitcoinBlockchainObjectModel));
}
return _er;
}
protected override void OnCreated(EventArgs e)
{
base.OnCreated(e);
OnObjectModelChanged(e);
}
protected override void OnObjectModelChanged(EventArgs e)
{
base.OnObjectModelChanged(e);
BitcoinBlockchainObjectModel bc = (ObjectModel as BitcoinBlockchainObjectModel);
if (bc == null) return;
if (!IsCreated) return;
for (int i = 0; i < Math.Min(6500, bc.Blocks.Count); i++)
{
tv.Model.Rows.Add(new TreeModelRow(new TreeModelRowColumn[]
{
new TreeModelRowColumn(tv.Model.Columns[0], bc.Blocks[i].Version.ToString()),
new TreeModelRowColumn(tv.Model.Columns[1], bc.Blocks[i].PreviousBlockHash.ToString()),
new TreeModelRowColumn(tv.Model.Columns[2], bc.Blocks[i].MerkelRoot.ToString()),
new TreeModelRowColumn(tv.Model.Columns[3], bc.Blocks[i].Timestamp.ToString())
}));
}
}
public override void UpdateSelections()
{
}
protected override Selection CreateSelectionInternal(object content)
{
return null;
}
}
}

View File

@ -0,0 +1,46 @@
//
// AssemblyInfo.cs
//
// Author:
// Michael Becker <alcexhim@gmail.com>
//
// Copyright (c) 2020 Mike Becker's Software
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
using System.Reflection;
using System.Runtime.CompilerServices;
// Information about this assembly is defined by the following attributes.
// Change them to the values specific to your project.
[assembly: AssemblyTitle("UniversalEditor.Plugins.Blockchain.UserInterface")]
[assembly: AssemblyDescription("")]
[assembly: AssemblyConfiguration("")]
[assembly: AssemblyCompany("Mike Becker's Software")]
[assembly: AssemblyProduct("")]
[assembly: AssemblyCopyright("Mike Becker's Software")]
[assembly: AssemblyTrademark("")]
[assembly: AssemblyCulture("")]
// The assembly version has the format "{Major}.{Minor}.{Build}.{Revision}".
// The form "{Major}.{Minor}.*" will automatically update the build and revision,
// and "{Major}.{Minor}.{Build}.*" will update just the revision.
[assembly: AssemblyVersion("1.0.*")]
// The following attributes are used to specify the signing key for the assembly,
// if desired. See the Mono documentation for more information about signing.
//[assembly: AssemblyDelaySign(false)]
//[assembly: AssemblyKeyFile("")]

View File

@ -0,0 +1,60 @@
<?xml version="1.0" encoding="utf-8"?>
<Project DefaultTargets="Build" ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<PropertyGroup>
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
<ProjectGuid>{B6E600F5-E5BC-4DC2-8B41-7B11EB0A11B3}</ProjectGuid>
<OutputType>Library</OutputType>
<RootNamespace>UniversalEditor.Plugins.Blockchain.UserInterface</RootNamespace>
<AssemblyName>UniversalEditor.Plugins.Blockchain.UserInterface</AssemblyName>
<TargetFrameworkVersion>v4.7</TargetFrameworkVersion>
<ReleaseVersion>4.0.2019.12</ReleaseVersion>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
<DebugSymbols>true</DebugSymbols>
<DebugType>full</DebugType>
<Optimize>false</Optimize>
<OutputPath>..\..\Output\Debug\Plugins</OutputPath>
<DefineConstants>DEBUG;</DefineConstants>
<ErrorReport>prompt</ErrorReport>
<WarningLevel>4</WarningLevel>
<ConsolePause>false</ConsolePause>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
<Optimize>true</Optimize>
<OutputPath>..\..\Output\Release\Plugins</OutputPath>
<ErrorReport>prompt</ErrorReport>
<WarningLevel>4</WarningLevel>
<ConsolePause>false</ConsolePause>
</PropertyGroup>
<ItemGroup>
<Reference Include="System" />
</ItemGroup>
<ItemGroup>
<Compile Include="BlockchainEditor.cs" />
<Compile Include="Properties\AssemblyInfo.cs" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\Libraries\UniversalEditor.Core\UniversalEditor.Core.csproj">
<Project>{2D4737E6-6D95-408A-90DB-8DFF38147E85}</Project>
<Name>UniversalEditor.Core</Name>
</ProjectReference>
<ProjectReference Include="..\..\Libraries\UniversalEditor.Essential\UniversalEditor.Essential.csproj">
<Project>{30467E5C-05BC-4856-AADC-13906EF4CADD}</Project>
<Name>UniversalEditor.Essential</Name>
</ProjectReference>
<ProjectReference Include="..\..\Libraries\UniversalEditor.UserInterface\UniversalEditor.UserInterface.csproj">
<Project>{8622EBC4-8E20-476E-B284-33D472081F5C}</Project>
<Name>UniversalEditor.UserInterface</Name>
</ProjectReference>
<ProjectReference Include="..\..\..\MBS.Framework\MBS.Framework\MBS.Framework.csproj">
<Project>{00266B21-35C9-4A7F-A6BA-D54D7FDCC25C}</Project>
<Name>MBS.Framework</Name>
</ProjectReference>
<ProjectReference Include="..\..\..\MBS.Framework.UserInterface\Libraries\MBS.Framework.UserInterface\MBS.Framework.UserInterface.csproj">
<Project>{29E1C1BB-3EA5-4062-B62F-85EEC703FE07}</Project>
<Name>MBS.Framework.UserInterface</Name>
</ProjectReference>
</ItemGroup>
<Import Project="$(MSBuildBinPath)\Microsoft.CSharp.targets" />
</Project>

View File

@ -0,0 +1,22 @@
<?xml version="1.0" encoding="utf-8" ?>
<UniversalEditor Version="4.0">
<Associations>
<Association>
<Filters>
<Filter Title="Bitcoin blockchain data">
<MagicByteSequences>
<MagicByteSequence>
<MagicByte Type="HexString">F9BEB4D9</MagicByte>
</MagicByteSequence>
</MagicByteSequences>
</Filter>
</Filters>
<ObjectModels>
<ObjectModel TypeName="UniversalEditor.Plugins.Blockchain.Bitcoin.ObjectModels.BitcoinBlockchainObjectModel" />
</ObjectModels>
<DataFormats>
<DataFormat TypeName="UniversalEditor.Plugins.Blockchain.Bitcoin.DataFormats.BitcoinBlockchainDataFormat" />
</DataFormats>
</Association>
</Associations>
</UniversalEditor>

View File

@ -0,0 +1,98 @@
using System;
using UniversalEditor.IO;
using UniversalEditor.Plugins.Blockchain.Bitcoin.ObjectModels;
namespace UniversalEditor.Plugins.Blockchain.Bitcoin.DataFormats
{
public class BitcoinBlockchainDataFormat : DataFormat
{
private static DataFormatReference _dfr = null;
protected override DataFormatReference MakeReferenceInternal()
{
if (_dfr == null)
{
_dfr = base.MakeReferenceInternal();
_dfr.Capabilities.Add(typeof(BitcoinBlockchainObjectModel), DataFormatCapabilities.All);
}
return _dfr;
}
public int HashLength { get; set; } = 32;
private readonly DateTime UNIX_EPOCH = new DateTime(1970, 01, 01, 00, 00, 00);
protected override void LoadInternal(ref ObjectModel objectModel)
{
BitcoinBlockchainObjectModel blockchain = (objectModel as BitcoinBlockchainObjectModel);
if (blockchain == null)
throw new ObjectModelNotSupportedException();
Reader reader = Accessor.Reader;
while (!reader.EndOfStream)
{
uint signature = reader.ReadUInt32();
if (signature != 0xD9B4BEF9)
throw new InvalidDataFormatException("file does not begin with 0xD9B4BEF9");
BitcoinBlock block = new BitcoinBlock();
uint datasize = reader.ReadUInt32();
block.Version = reader.ReadUInt32();
block.PreviousBlockHash = reader.ReadBytes(HashLength);
block.MerkelRoot = reader.ReadBytes(HashLength);
uint timestamp = reader.ReadUInt32();
block.Timestamp = UNIX_EPOCH.AddSeconds(timestamp);
block.Bits = reader.ReadUInt32();
block.Nonce = reader.ReadUInt32();
ulong transactionCount = ReadCompactNumber(reader);
for (ulong i = 0; i < transactionCount; i++)
{
BitcoinBlockTransaction transaction = new BitcoinBlockTransaction();
uint transactionVersion = reader.ReadUInt32();
ulong inputs = ReadCompactNumber(reader);
for (ulong j = 0; j < inputs; j++)
{
byte[] previousOutput = reader.ReadBytes(HashLength);
uint previousOutput2 = reader.ReadUInt32();
ulong scriptLength = ReadCompactNumber(reader);
byte[] scriptData = reader.ReadBytes(scriptLength);
uint sequence = reader.ReadUInt32();
}
byte outputs = reader.ReadByte();
for (int j = 0; j < outputs; j++)
{
ulong output_c = reader.ReadUInt64();
ulong publicKeyScriptLength = ReadCompactNumber(reader);
transaction.PublicKeyScript = reader.ReadBytes(publicKeyScriptLength);
}
uint locktime = reader.ReadUInt32();
block.Transactions.Add(transaction);
}
blockchain.Blocks.Add(block);
}
}
private ulong ReadCompactNumber(Reader reader)
{
byte num = reader.ReadByte();
if (num == 0xFD)
{
return reader.ReadUInt16();
}
else if (num == 0xFE)
{
return reader.ReadUInt32();
}
else if (num == 0xFF)
{
return reader.ReadUInt64();
}
return num;
}
protected override void SaveInternal(ObjectModel objectModel)
{
throw new NotImplementedException();
}
}
}

View File

@ -0,0 +1,27 @@
using System;
namespace UniversalEditor.Plugins.Blockchain.Bitcoin.ObjectModels
{
public class BitcoinBlock : Block
{
public class BitcoinBlockCollection
: System.Collections.ObjectModel.Collection<BitcoinBlock>
{
}
public uint Version { get; internal set; }
public byte[] PreviousBlockHash { get; set; }
public byte[] MerkelRoot { get; set; }
public DateTime Timestamp { get; set; }
public uint Bits { get; set; }
public uint Nonce { get; set; }
public BitcoinBlockTransaction.BitcoinBlockTransactionCollection Transactions { get; } = new BitcoinBlockTransaction.BitcoinBlockTransactionCollection();
public override object Clone()
{
BitcoinBlock clone = new BitcoinBlock();
clone.Version = Version;
return clone;
}
}
}

View File

@ -0,0 +1,20 @@
using System;
namespace UniversalEditor.Plugins.Blockchain.Bitcoin.ObjectModels
{
public class BitcoinBlockTransaction : BlockTransaction
{
public byte[] PublicKeyScript { get; set; }
public class BitcoinBlockTransactionCollection
: System.Collections.ObjectModel.Collection<BitcoinBlockTransaction>
{
}
public override object Clone()
{
BitcoinBlockTransaction clone = new BitcoinBlockTransaction();
return clone;
}
}
}

View File

@ -0,0 +1,36 @@
using System;
namespace UniversalEditor.Plugins.Blockchain.Bitcoin.ObjectModels
{
public class BitcoinBlockchainObjectModel : ObjectModel
{
private static ObjectModelReference _omr = null;
protected override ObjectModelReference MakeReferenceInternal()
{
if (_omr == null)
{
_omr = base.MakeReferenceInternal();
_omr.Path = new string[] { "Blockchain", "Bitcoin" };
}
return _omr;
}
public BitcoinBlock.BitcoinBlockCollection Blocks { get; } = new BitcoinBlock.BitcoinBlockCollection();
public override void Clear()
{
Blocks.Clear();
}
public override void CopyTo(ObjectModel where)
{
BitcoinBlockchainObjectModel clone = (where as BitcoinBlockchainObjectModel);
if (clone == null)
throw new ObjectModelNotSupportedException();
for (int i = 0; i < Blocks.Count;i++)
{
clone.Blocks.Add(Blocks[i].Clone() as BitcoinBlock);
}
}
}
}

View File

@ -0,0 +1,8 @@
using System;
namespace UniversalEditor.Plugins.Blockchain
{
public abstract class Block : ICloneable
{
public abstract object Clone();
}
}

View File

@ -0,0 +1,8 @@
using System;
namespace UniversalEditor.Plugins.Blockchain
{
public abstract class BlockTransaction : ICloneable
{
public abstract object Clone();
}
}

View File

@ -0,0 +1,26 @@
using System.Reflection;
using System.Runtime.CompilerServices;
// Information about this assembly is defined by the following attributes.
// Change them to the values specific to your project.
[assembly: AssemblyTitle("UniversalEditor.Plugins.Blockchain")]
[assembly: AssemblyDescription("")]
[assembly: AssemblyConfiguration("")]
[assembly: AssemblyCompany("Mike Becker's Software")]
[assembly: AssemblyProduct("")]
[assembly: AssemblyCopyright("Mike Becker's Software")]
[assembly: AssemblyTrademark("")]
[assembly: AssemblyCulture("")]
// The assembly version has the format "{Major}.{Minor}.{Build}.{Revision}".
// The form "{Major}.{Minor}.*" will automatically update the build and revision,
// and "{Major}.{Minor}.{Build}.*" will update just the revision.
[assembly: AssemblyVersion("1.0.*")]
// The following attributes are used to specify the signing key for the assembly,
// if desired. See the Mono documentation for more information about signing.
//[assembly: AssemblyDelaySign(false)]
//[assembly: AssemblyKeyFile("")]

View File

@ -0,0 +1,58 @@
<?xml version="1.0" encoding="utf-8"?>
<Project DefaultTargets="Build" ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<PropertyGroup>
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
<ProjectGuid>{C54F6BCD-60CD-4603-B0C9-CD0864455CB1}</ProjectGuid>
<OutputType>Library</OutputType>
<RootNamespace>UniversalEditor.Plugins.Blockchain</RootNamespace>
<AssemblyName>UniversalEditor.Plugins.Blockchain</AssemblyName>
<TargetFrameworkVersion>v4.7</TargetFrameworkVersion>
<ReleaseVersion>4.0.2019.12</ReleaseVersion>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
<DebugSymbols>true</DebugSymbols>
<DebugType>full</DebugType>
<Optimize>false</Optimize>
<OutputPath>..\..\Output\Debug\Plugins</OutputPath>
<DefineConstants>DEBUG;</DefineConstants>
<ErrorReport>prompt</ErrorReport>
<WarningLevel>4</WarningLevel>
<ConsolePause>false</ConsolePause>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
<Optimize>true</Optimize>
<OutputPath>..\..\Output\Release\Plugins</OutputPath>
<ErrorReport>prompt</ErrorReport>
<WarningLevel>4</WarningLevel>
<ConsolePause>false</ConsolePause>
</PropertyGroup>
<ItemGroup>
<Reference Include="System" />
</ItemGroup>
<ItemGroup>
<Compile Include="BlockTransaction.cs" />
<Compile Include="Properties\AssemblyInfo.cs" />
<Compile Include="Bitcoin\DataFormats\BitcoinBlockchainDataFormat.cs" />
<Compile Include="Bitcoin\ObjectModels\BitcoinBlock.cs" />
<Compile Include="Bitcoin\ObjectModels\BitcoinBlockchainObjectModel.cs" />
<Compile Include="Bitcoin\ObjectModels\BitcoinBlockTransaction.cs" />
<Compile Include="Block.cs" />
</ItemGroup>
<ItemGroup>
<Folder Include="Bitcoin\" />
<Folder Include="Bitcoin\DataFormats\" />
<Folder Include="Bitcoin\ObjectModels\" />
<Folder Include="Associations\" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\Libraries\UniversalEditor.Core\UniversalEditor.Core.csproj">
<Project>{2D4737E6-6D95-408A-90DB-8DFF38147E85}</Project>
<Name>UniversalEditor.Core</Name>
</ProjectReference>
</ItemGroup>
<ItemGroup>
<EmbeddedResource Include="Associations\BitcoinBlockchain.uexml" />
</ItemGroup>
<Import Project="$(MSBuildBinPath)\Microsoft.CSharp.targets" />
</Project>

View File

@ -201,6 +201,10 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Content", "Content", "{63E2
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "MBS.Framework.UserInterface.Content.PlatformIndependent", "..\MBS.Framework.UserInterface\Content\MBS.Framework.UserInterface.Content.PlatformIndependent\MBS.Framework.UserInterface.Content.PlatformIndependent.csproj", "{FAE48F29-DB35-4CD6-8A55-6C1FDDFBE6AF}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UniversalEditor.Plugins.Blockchain", "Plugins\UniversalEditor.Plugins.Blockchain\UniversalEditor.Plugins.Blockchain.csproj", "{C54F6BCD-60CD-4603-B0C9-CD0864455CB1}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UniversalEditor.Plugins.Blockchain.UserInterface", "Plugins.UserInterface\UniversalEditor.Plugins.Blockchain.UserInterface\UniversalEditor.Plugins.Blockchain.UserInterface.csproj", "{B6E600F5-E5BC-4DC2-8B41-7B11EB0A11B3}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
@ -577,6 +581,14 @@ Global
{FAE48F29-DB35-4CD6-8A55-6C1FDDFBE6AF}.Debug|Any CPU.Build.0 = Debug|Any CPU
{FAE48F29-DB35-4CD6-8A55-6C1FDDFBE6AF}.Release|Any CPU.ActiveCfg = Release|Any CPU
{FAE48F29-DB35-4CD6-8A55-6C1FDDFBE6AF}.Release|Any CPU.Build.0 = Release|Any CPU
{C54F6BCD-60CD-4603-B0C9-CD0864455CB1}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{C54F6BCD-60CD-4603-B0C9-CD0864455CB1}.Debug|Any CPU.Build.0 = Debug|Any CPU
{C54F6BCD-60CD-4603-B0C9-CD0864455CB1}.Release|Any CPU.ActiveCfg = Release|Any CPU
{C54F6BCD-60CD-4603-B0C9-CD0864455CB1}.Release|Any CPU.Build.0 = Release|Any CPU
{B6E600F5-E5BC-4DC2-8B41-7B11EB0A11B3}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{B6E600F5-E5BC-4DC2-8B41-7B11EB0A11B3}.Debug|Any CPU.Build.0 = Debug|Any CPU
{B6E600F5-E5BC-4DC2-8B41-7B11EB0A11B3}.Release|Any CPU.ActiveCfg = Release|Any CPU
{B6E600F5-E5BC-4DC2-8B41-7B11EB0A11B3}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(NestedProjects) = preSolution
{6F0AB1AF-E1A1-4D19-B19C-05BBB15C94B2} = {05D15661-E684-4EC9-8FBD-C014BA433CC5}
@ -671,6 +683,8 @@ Global
{D1FB19C4-025E-4D4A-8532-4196AFCC8813} = {5E4765D1-3959-4433-8E9C-992E26D7BE62}
{63E2F10F-27A6-4BAA-BF4A-4422D0934E91} = {20F315E0-52AE-479F-AF43-3402482C1FC8}
{FAE48F29-DB35-4CD6-8A55-6C1FDDFBE6AF} = {63E2F10F-27A6-4BAA-BF4A-4422D0934E91}
{C54F6BCD-60CD-4603-B0C9-CD0864455CB1} = {2ED32D16-6C06-4450-909A-40D32DA67FB4}
{B6E600F5-E5BC-4DC2-8B41-7B11EB0A11B3} = {7B535D74-5496-4802-B809-89ED88274A91}
EndGlobalSection
GlobalSection(MonoDevelopProperties) = preSolution
Policies = $0