Browse Source

Add LND support

migrate-pregen
nicolas.dorier 7 years ago
parent
commit
9e58d685cb
  1. 2
      docker-compose-generator/docker-fragments/bitcoin-clightning.yml
  2. 49
      docker-compose-generator/docker-fragments/bitcoin-lnd.yml
  3. 2
      docker-compose-generator/docker-fragments/btcpayserver.yml
  4. 2
      docker-compose-generator/docker-fragments/litecoin-clightning.yml
  5. 49
      docker-compose-generator/docker-fragments/litecoin-lnd.yml
  6. 20
      docker-compose-generator/src/CryptoDefinition.cs
  7. 247
      docker-compose-generator/src/DockerComposeDefinition.cs
  8. 6
      docker-compose-generator/src/Program.cs

2
docker-compose-generator/docker-fragments/bitcoin-clightning.yml

@ -23,7 +23,7 @@ services:
- bitcoind - bitcoind
btcpayserver: btcpayserver:
environment: environment:
BTCPAY_BTCLIGHTNING: "/etc/clightning_bitcoin/lightning-rpc" BTCPAY_BTCLIGHTNING: "type=clightning;server=unix://etc/clightning_bitcoin/lightning-rpc"
volumes: volumes:
- "clightning_bitcoin_datadir:/etc/clightning_bitcoin" - "clightning_bitcoin_datadir:/etc/clightning_bitcoin"
links: links:

49
docker-compose-generator/docker-fragments/bitcoin-lnd.yml

@ -0,0 +1,49 @@
version: "3"
services:
lnd_bitcoin:
image: btcpayserver/lnd:0.4.2.0
container_name: btcpayserver_lnd_bitcoin
environment:
LND_CHAIN: "btc"
LND_ENVIRONMENT: "${NBITCOIN_NETWORK:-regtest}"
LND_EXTRA_ARGS: |
restlisten=0.0.0.0:8080
bitcoin.node=bitcoind
bitcoind.rpchost=bitcoind:43782
bitcoind.zmqpath=tcp://bitcoind:28332
externalip=${BTCPAY_HOST}:9735
alias=${LIGHTNING_ALIAS}
noencryptwallet=1
notls=1
ports:
- "9735:9735"
expose:
- "8080"
- "9735"
volumes:
- "lnd_bitcoin_datadir:/data"
- "bitcoin_datadir:/deps/.bitcoin"
links:
- bitcoind
btcpayserver:
environment:
BTCPAY_BTCLIGHTNING: "type=lnd-rest;server=http://lnd_bitcoin:8080/;macaroonfilepath=/etc/lnd_bitcoin/admin.macaroon;allowinsecure=true"
volumes:
- "lnd_bitcoin_datadir:/etc/lnd_bitcoin"
links:
- lnd_bitcoin
bitcoind:
environment:
BITCOIN_EXTRA_ARGS: |
zmqpubrawtx=tcp://0.0.0.0:28332
zmqpubrawblock=tcp://0.0.0.0:28332
zmqpubrawtxlock=tcp://0.0.0.0:28332
zmqpubhashblock=tcp://0.0.0.0:28332
expose:
- "28332"
volumes:
lnd_bitcoin_datadir:

2
docker-compose-generator/docker-fragments/btcpayserver.yml

@ -4,7 +4,7 @@ services:
btcpayserver: btcpayserver:
restart: unless-stopped restart: unless-stopped
image: nicolasdorier/btcpayserver:1.0.2.39 image: nicolasdorier/btcpayserver:1.0.2.40
expose: expose:
- "49392" - "49392"
environment: environment:

2
docker-compose-generator/docker-fragments/litecoin-clightning.yml

@ -23,7 +23,7 @@ services:
- litecoind - litecoind
btcpayserver: btcpayserver:
environment: environment:
BTCPAY_LTCLIGHTNING: "/etc/clightning_litecoin/lightning-rpc" BTCPAY_LTCLIGHTNING: "type=clightning;server=unix://etc/clightning_litecoin/lightning-rpc"
volumes: volumes:
- "clightning_litecoin_datadir:/etc/clightning_litecoin" - "clightning_litecoin_datadir:/etc/clightning_litecoin"
links: links:

49
docker-compose-generator/docker-fragments/litecoin-lnd.yml

@ -0,0 +1,49 @@
version: "3"
services:
lnd_litecoin:
image: btcpayserver/lnd:0.4.2.0
container_name: btcpayserver_lnd_litecoin
environment:
LND_CHAIN: "ltc"
LND_ENVIRONMENT: "${NBITCOIN_NETWORK:-regtest}"
LND_EXTRA_ARGS: |
restlisten=0.0.0.0:8080
litecoin.node=litecoind
litecoind.rpchost=litecoind:43782
litecoind.zmqpath=tcp://litecoind:28332
externalip=${BTCPAY_HOST}:9736
alias=${LIGHTNING_ALIAS}
noencryptwallet=1
notls=1
ports:
- "9736:9735"
expose:
- "8080"
- "9736"
volumes:
- "lnd_litecoin_datadir:/data"
- "litecoin_datadir:/deps/.litecoin"
links:
- litecoind
btcpayserver:
environment:
BTCPAY_LTCLIGHTNING: "type=lnd-rest;server=http://lnd_litecoin:8080/;macaroonfilepath=/etc/lnd_litecoin/admin.macaroon;allowinsecure=true"
volumes:
- "lnd_litecoin_datadir:/etc/lnd_litecoin"
links:
- lnd_litecoin
litecoind:
environment:
BITCOIN_EXTRA_ARGS: |
zmqpubrawtx=tcp://0.0.0.0:28332
zmqpubrawblock=tcp://0.0.0.0:28332
zmqpubrawtxlock=tcp://0.0.0.0:28332
zmqpubhashblock=tcp://0.0.0.0:28332
expose:
- "28332"
volumes:
lnd_litecoin_datadir:

20
docker-compose-generator/src/CryptoDefinition.cs

@ -21,8 +21,13 @@ namespace DockerGenerator
get; get;
private set; private set;
} }
public string LNDFragment
{
get;
private set;
}
public static CryptoDefinition[] GetDefinitions() public static CryptoDefinition[] GetDefinitions()
{ {
return new[] return new[]
{ {
@ -31,30 +36,29 @@ namespace DockerGenerator
Crypto = "ltc", Crypto = "ltc",
CryptoFragment = "litecoin", CryptoFragment = "litecoin",
CLightningFragment = "litecoin-clightning", CLightningFragment = "litecoin-clightning",
}, LNDFragment = "litecoin-lnd"
},
new CryptoDefinition() new CryptoDefinition()
{ {
Crypto = "btc", Crypto = "btc",
CryptoFragment = "bitcoin", CryptoFragment = "bitcoin",
CLightningFragment = "bitcoin-clightning", CLightningFragment = "bitcoin-clightning",
LNDFragment = "bitcoin-lnd"
}, },
new CryptoDefinition() new CryptoDefinition()
{ {
Crypto = "btg", Crypto = "btg",
CryptoFragment = "bgold", CryptoFragment = "bgold"
CLightningFragment = null,
}, },
new CryptoDefinition() new CryptoDefinition()
{ {
Crypto = "ftc", Crypto = "ftc",
CryptoFragment = "feathercoin", CryptoFragment = "feathercoin"
CLightningFragment = null,
}, },
new CryptoDefinition() new CryptoDefinition()
{ {
Crypto = "grs", Crypto = "grs",
CryptoFragment = "groestlcoin", CryptoFragment = "groestlcoin"
CLightningFragment = null,
} }
}; };
} }

247
docker-compose-generator/src/DockerComposeDefinition.cs

@ -8,128 +8,145 @@ using System.IO;
namespace DockerGenerator namespace DockerGenerator
{ {
public class DockerComposeDefinition public class DockerComposeDefinition
{ {
public List<string> Fragments public List<string> Fragments
{ {
get; set; get; set;
} }
private string _Name; private string _Name;
public DockerComposeDefinition(string name, List<string> fragments) public DockerComposeDefinition(string name, List<string> fragments)
{ {
Fragments = fragments; Fragments = fragments;
_Name = name; _Name = name;
} }
public string FragmentLocation public string FragmentLocation
{ {
get; set; get; set;
} }
public string BuildOutputDirectory public string BuildOutputDirectory
{ {
get; set; get; set;
} }
public string GetFilePath() public string GetFilePath()
{ {
return Path.Combine(BuildOutputDirectory, $"docker-compose.{_Name}.yml"); return Path.Combine(BuildOutputDirectory, $"docker-compose.{_Name}.yml");
} }
public void Build() public void Build()
{ {
Console.WriteLine($"Generating {GetFilePath()}"); Console.WriteLine($"Generating {GetFilePath()}");
var deserializer = new DeserializerBuilder().Build(); var deserializer = new DeserializerBuilder().Build();
var serializer = new SerializerBuilder().Build(); var serializer = new SerializerBuilder().Build();
Console.WriteLine($"With fragments:"); Console.WriteLine($"With fragments:");
foreach(var fragment in Fragments) foreach(var fragment in Fragments)
{ {
Console.WriteLine($"\t{fragment}"); Console.WriteLine($"\t{fragment}");
} }
var services = new List<KeyValuePair<YamlNode, YamlNode>>(); var services = new List<KeyValuePair<YamlNode, YamlNode>>();
var volumes = new List<KeyValuePair<YamlNode, YamlNode>>(); var volumes = new List<KeyValuePair<YamlNode, YamlNode>>();
foreach(var doc in Fragments.Select(f => ParseDocument(f))) foreach(var doc in Fragments.Select(f => ParseDocument(f)))
{ {
if(doc.Children.ContainsKey("services") && doc.Children["services"] is YamlMappingNode fragmentServicesRoot) if(doc.Children.ContainsKey("services") && doc.Children["services"] is YamlMappingNode fragmentServicesRoot)
{ {
services.AddRange(fragmentServicesRoot.Children); services.AddRange(fragmentServicesRoot.Children);
} }
if(doc.Children.ContainsKey("volumes") && doc.Children["volumes"] is YamlMappingNode fragmentVolumesRoot) if(doc.Children.ContainsKey("volumes") && doc.Children["volumes"] is YamlMappingNode fragmentVolumesRoot)
{ {
volumes.AddRange(fragmentVolumesRoot.Children); volumes.AddRange(fragmentVolumesRoot.Children);
} }
} }
YamlMappingNode output = new YamlMappingNode();
output.Add("version", new YamlScalarNode("3") { Style = YamlDotNet.Core.ScalarStyle.DoubleQuoted });
output.Add("services", new YamlMappingNode(Merge(services)));
output.Add("volumes", new YamlMappingNode(volumes));
var result = serializer.Serialize(output);
var outputFile = GetFilePath();
File.WriteAllText(outputFile, result.Replace("''", ""));
Console.WriteLine($"Generated {outputFile}");
Console.WriteLine();
}
private KeyValuePair<YamlNode, YamlNode>[] Merge(List<KeyValuePair<YamlNode, YamlNode>> services) YamlMappingNode output = new YamlMappingNode();
{ output.Add("version", new YamlScalarNode("3") { Style = YamlDotNet.Core.ScalarStyle.DoubleQuoted });
return services output.Add("services", new YamlMappingNode(Merge(services)));
.GroupBy(s => s.Key.ToString(), s=> s.Value) output.Add("volumes", new YamlMappingNode(volumes));
.Select(group => var result = serializer.Serialize(output);
(GroupName: group.Key, var outputFile = GetFilePath();
MainNode: group.OfType<YamlMappingNode>().Single(n=> n.Children.ContainsKey("image")), File.WriteAllText(outputFile, result.Replace("''", ""));
MergedNodes: group.OfType<YamlMappingNode>().Where(n => !n.Children.ContainsKey("image")))) Console.WriteLine($"Generated {outputFile}");
.Select(_ => Console.WriteLine();
{ }
foreach(var node in _.MergedNodes)
{
foreach(var child in node)
{
var childValue = child.Value;
if(!_.MainNode.Children.TryGetValue(child.Key, out var mainChildValue))
{
mainChildValue = child.Value;
_.MainNode.Add(child.Key, child.Value);
}
else if(childValue is YamlMappingNode childMapping && mainChildValue is YamlMappingNode mainChildMapping)
{
foreach(var leaf in childMapping)
{
if(mainChildMapping.Children.TryGetValue(leaf.Key, out var mainLeaf))
{
if(leaf.Value is YamlScalarNode leafScalar && mainLeaf is YamlScalarNode leafMainScalar)
{
leafMainScalar.Value = leafMainScalar.Value + "," + leaf.Value;
}
}
else
{
mainChildMapping.Add(leaf.Key, leaf.Value);
}
}
}
else if(childValue is YamlSequenceNode childSequence && mainChildValue is YamlSequenceNode mainSequence)
{
foreach(var c in childSequence.Children)
{
mainSequence.Add(c);
}
}
}
}
return new KeyValuePair<YamlNode, YamlNode>(_.GroupName, _.MainNode);
}).ToArray();
}
private YamlMappingNode ParseDocument(string fragment) private KeyValuePair<YamlNode, YamlNode>[] Merge(List<KeyValuePair<YamlNode, YamlNode>> services)
{ {
var input = new StringReader(File.ReadAllText(Path.Combine(FragmentLocation, $"{fragment}.yml"))); return services
YamlStream stream = new YamlStream(); .GroupBy(s => s.Key.ToString(), s => s.Value)
stream.Load(input); .Select(group =>
return (YamlMappingNode)stream.Documents[0].RootNode; (GroupName: group.Key,
} MainNode: group.OfType<YamlMappingNode>().Single(n => n.Children.ContainsKey("image")),
} MergedNodes: group.OfType<YamlMappingNode>().Where(n => !n.Children.ContainsKey("image"))))
.Select(_ =>
{
foreach(var node in _.MergedNodes)
{
foreach(var child in node)
{
var childValue = child.Value;
if(!_.MainNode.Children.TryGetValue(child.Key, out var mainChildValue))
{
mainChildValue = child.Value;
_.MainNode.Add(child.Key, child.Value);
}
else if(childValue is YamlMappingNode childMapping && mainChildValue is YamlMappingNode mainChildMapping)
{
foreach(var leaf in childMapping)
{
if(mainChildMapping.Children.TryGetValue(leaf.Key, out var mainLeaf))
{
if(leaf.Value is YamlScalarNode leafScalar && mainLeaf is YamlScalarNode leafMainScalar)
{
var eof = EOF(leafMainScalar.Value) ?? EOF(leaf.Value.ToString());
if(eof != null)
{
leafMainScalar.Value = leafMainScalar.Value + eof + leaf.Value;
}
else
{
leafMainScalar.Value = leafMainScalar.Value + "," + leaf.Value;
}
}
}
else
{
mainChildMapping.Add(leaf.Key, leaf.Value);
}
}
}
else if(childValue is YamlSequenceNode childSequence && mainChildValue is YamlSequenceNode mainSequence)
{
foreach(var c in childSequence.Children)
{
mainSequence.Add(c);
}
}
}
}
return new KeyValuePair<YamlNode, YamlNode>(_.GroupName, _.MainNode);
}).ToArray();
}
private string EOF(string value)
{
if(value.Contains("\r\n", StringComparison.OrdinalIgnoreCase))
return "\r\n";
if(value.Contains("\n", StringComparison.OrdinalIgnoreCase))
return "\n";
return null;
}
private YamlMappingNode ParseDocument(string fragment)
{
var input = new StringReader(File.ReadAllText(Path.Combine(FragmentLocation, $"{fragment}.yml")));
YamlStream stream = new YamlStream();
stream.Load(input);
return (YamlMappingNode)stream.Documents[0].RootNode;
}
}
} }

6
docker-compose-generator/src/Program.cs

@ -86,7 +86,11 @@ namespace DockerGenerator
{ {
fragments.Add(crypto.CLightningFragment); fragments.Add(crypto.CLightningFragment);
} }
} if(composition.SelectedLN == "lnd" && crypto.LNDFragment != null)
{
fragments.Add(crypto.LNDFragment);
}
}
var def = new DockerComposeDefinition(name, fragments); var def = new DockerComposeDefinition(name, fragments);
def.FragmentLocation = fragmentLocation; def.FragmentLocation = fragmentLocation;

Loading…
Cancel
Save