artq 0.0.1 → 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: e392049f9000bfe8a6b16ee2cad088a5fc7d2d4d76e0be088522ee6839e00822
4
- data.tar.gz: '0904cfde265b254156d8a52cafbd6999a016d75e5ccad53a5bb5c1312efaae85'
3
+ metadata.gz: c9779a590a436cc490adf3f14a21c2a663c349a56cc2d3546fa3d7ca490ed157
4
+ data.tar.gz: 8c54bc4cdbc6e1626f13f5d445b39a0fa6d029f9330aefa030f37d7f8fa83558
5
5
  SHA512:
6
- metadata.gz: dc71ccf9a977d4764218f3681ee72b3a34c3cd00638768c905c9037c77922fa4c3ad8dd154e195565b3d8d9b4e079b4b36cdda340036eb4d58f7456e668bfaa2
7
- data.tar.gz: cf8653c6642afe656cad20b63e03b3689a315a246490dd6d7d153518f10f1269382a4a31d6b8eff0ed646bbc9828d538485730d52adb08d81597c9c19d1a5ba9
6
+ metadata.gz: c75af5b52d7b37b752f9e7de03b5e3d7f84f8ead09cc3f537c1897c2468990a812b2de9b6fd2a44c1c54237bf6cc4120e63ea0b31b1c4bd907db58c56412ef82
7
+ data.tar.gz: d37f4a40864b66dccf69b629f07572f10abb38036047bd35e04947499f7de767800f87f8fe852414b9201835782589a8e6d237a60748d36d22d558140b2f763c
data/Manifest.txt CHANGED
@@ -2,5 +2,6 @@ CHANGELOG.md
2
2
  Manifest.txt
3
3
  README.md
4
4
  Rakefile
5
+ bin/artq
5
6
  lib/artq.rb
6
7
  lib/artq/version.rb
data/README.md CHANGED
@@ -1,6 +1,6 @@
1
1
  # ArtQ
2
2
 
3
- artq - query (ethereum) blockchain contracts / services for data about art collections via json-rpc
3
+ artq - query (ethereum) blockchain contracts / services for (meta) data about art collections via json-rpc
4
4
 
5
5
 
6
6
  * home :: [github.com/pixelartexchange/artbase](https://github.com/pixelartexchange/artbase)
@@ -10,9 +10,178 @@ artq - query (ethereum) blockchain contracts / services for data about art colle
10
10
 
11
11
 
12
12
 
13
+
13
14
  ## Usage
14
15
 
15
- To be done
16
+
17
+ ### Step 0: Setup JSON RPC Client
18
+
19
+ The ArtQ command line tool
20
+ gets the eth node uri via the INFURA_URI enviroment variable / key for now.
21
+ Set the environment variable / key
22
+ depending on your operating system (OS) e.g.:
23
+
24
+ ```
25
+ set INFURA_URI=https://mainnet.infura.io/v3/<YOUR_KEY_HERE>
26
+ ```
27
+
28
+
29
+ ### Query (Token) Collection Info
30
+
31
+ To use the artq command line tool pass in the art collection contract address in the hex (string) format.
32
+
33
+
34
+ #### "Off-Blockchain" Token Metadata - Case No. 1
35
+
36
+ Let's try Moonbirds - an ("off-blockchain") pixel art collection -
37
+ with the token contract / service at [0x23581767a106ae21c074b2276d25e5c3e136a68b](https://etherscan.io/address/0x23581767a106ae21c074b2276d25e5c3e136a68b):
38
+
39
+ ```
40
+ $ artq 0x23581767a106ae21c074b2276d25e5c3e136a68b
41
+ ```
42
+
43
+ resulting in:
44
+
45
+ ```
46
+ name: >Moonbirds<
47
+ symbol: >MOONBIRD<
48
+ totalSupply: >10000<
49
+
50
+ tokenURIs 0..2:
51
+ tokenId: 0
52
+ https://live---metadata-5covpqijaa-uc.a.run.app/metadata/0
53
+ tokenId: 1
54
+ https://live---metadata-5covpqijaa-uc.a.run.app/metadata/1
55
+ tokenId: 2
56
+ https://live---metadata-5covpqijaa-uc.a.run.app/metadata/2
57
+ ```
58
+
59
+ Note: By default the tokenURI method gets called / queried
60
+ for the first tokens (e.g. 0, 1, 2, etc.).
61
+
62
+ If you get a link back (e.g. starting with `https://` or `ipfs://`)
63
+ than the art collection is "off-blockchain" and
64
+ you MUST follow / request the link to get the token metadata.
65
+
66
+
67
+ For example if you request <https://live---metadata-5covpqijaa-uc.a.run.app/metadata/0>
68
+ you get back:
69
+
70
+ ``` json
71
+ {"name":"#0",
72
+ "image":"https://live---metadata-5covpqijaa-uc.a.run.app/images/0",
73
+ "external_url":"https://moonbirds.xyz/",
74
+ "attributes":[
75
+ {"trait_type":"Eyes","value":"Angry"},
76
+ {"trait_type":"Outerwear","value":"Hoodie Down"},
77
+ {"trait_type":"Body","value":"Tabby"},
78
+ {"trait_type":"Feathers","value":"Gray"},
79
+ {"trait_type":"Background","value":"Green"},
80
+ {"trait_type":"Beak","value":"Small"}],
81
+ "x_debug":["orig:9650"]}
82
+ ```
83
+
84
+
85
+
86
+ #### "On-Blockchain" Token Metadata (With Inline Image) - Case No. 2
87
+
88
+
89
+
90
+ Let's try The Saudis - an ("on-blockchain") pixel art collection -
91
+ with the token contract / service at [0xe21ebcd28d37a67757b9bc7b290f4c4928a430b1](https://etherscan.io/address/0xe21ebcd28d37a67757b9bc7b290f4c4928a430b1):
92
+
93
+ ```
94
+ $ artq 0xe21ebcd28d37a67757b9bc7b290f4c4928a430b1
95
+ ```
96
+
97
+ resulting in:
98
+
99
+ ```
100
+ name: >The Saudis<
101
+ symbol: >SAUD<
102
+ totalSupply: >5555<
103
+
104
+ tokenURIs 0..2:
105
+ tokenId 0:
106
+ ```
107
+ ``` json
108
+ {"name":"The Saudis #0",
109
+ "description":"Max Bidding",
110
+ "image_data": "...",
111
+ "external_url":"https://token.thesaudisnft.com/0",
112
+ "attributes":
113
+ [{"trait_type":"Head", "value":"Light 1"},
114
+ {"trait_type":"Hair", "value":"Bald"},
115
+ {"trait_type":"Facial Hair", "value":"Normal Brown Beard & Mustache"},
116
+ {"trait_type":"Headwear", "value":"Haram Police Cap"},
117
+ {"trait_type":"Eyewear", "value":"Regular Pixel Shades"},
118
+ {"trait_type":"Mouthpiece", "value":"None"}]}
119
+ ```
120
+
121
+ ```
122
+ tokenId 1:
123
+ ```
124
+ ``` json
125
+ {"name":"The Saudis #1",
126
+ "description":"Max Bidding",
127
+ "image_data": "...",
128
+ "external_url":"https://token.thesaudisnft.com/1",
129
+ "attributes":
130
+ [{"trait_type":"Head", "value":"Light 1"},
131
+ {"trait_type":"Hair", "value":"Long Widow's Peak"},
132
+ {"trait_type":"Facial Hair", "value":"Messy White Beard"},
133
+ {"trait_type":"Headwear", "value":"Brown Shemagh & Agal"},
134
+ {"trait_type":"Eyewear", "value":"Big Purple Shades"},
135
+ {"trait_type":"Mouthpiece", "value":"None"}]}
136
+ ```
137
+
138
+ ```
139
+ tokenId 2:
140
+ ```
141
+ ``` json
142
+ {"name":"The Saudis #2 🛢" ,
143
+ "description":"Max Bidding",
144
+ "image_data": "...",
145
+ "external_url":"https://token.thesaudisnft.com/2",
146
+ "attributes":
147
+ [{"trait_type":"Head", "value":"Dark 1"},
148
+ {"trait_type":"Hair", "value":"Short Buzz Cut"},
149
+ {"trait_type":"Facial Hair", "value":"Gradient Beard"},
150
+ {"trait_type":"Headwear", "value":"Brown Shemagh & Agal"},
151
+ {"trait_type":"Eyewear", "value":"Big Pixel Shades"},
152
+ {"trait_type":"Mouthpiece", "value":"Shadowless Vape"}]}
153
+ ```
154
+
155
+ Note: The artq command-line tool "auto-magically"
156
+ decodes "on-blockchain" metadata in the base64 format
157
+ and inline svg images in the base64 format get "cut" from the metadata and "pasted" decoded. Example for tokenId 0, that is, The Saudis #0:
158
+
159
+ ``` xml
160
+ <svg xmlns="http://www.w3.org/2000/svg"
161
+ xmlns:xlink="http://www.w3.org/1999/xlink" image-rendering="pixelated"
162
+ height="336" width="336">
163
+ <foreignObject x="0" y="0" width="336" height="336">
164
+ <img xmlns="http://www.w3.org/1999/xhtml" height="336" width="336" src=""/>
165
+ </foreignObject>
166
+ <foreignObject x="0" y="0" width="336" height="336">
167
+ <img xmlns="http://www.w3.org/1999/xhtml" height="336" width="336" src="" />
168
+ </foreignObject>
169
+ <foreignObject x="0" y="0" width="336" height="336">
170
+ <img xmlns="http://www.w3.org/1999/xhtml" height="336" width="336" src="" />
171
+ </foreignObject>
172
+ <foreignObject x="0" y="0" width="336" height="336">
173
+ <img xmlns="http://www.w3.org/1999/xhtml" height="336" width="336" src="" />
174
+ </foreignObject>
175
+ <foreignObject x="0" y="0" width="336" height="336">
176
+ <img xmlns="http://www.w3.org/1999/xhtml" height="336" width="336" src="" />
177
+ </foreignObject>
178
+ <foreignObject x="0" y="0" width="336" height="336">
179
+ <img xmlns="http://www.w3.org/1999/xhtml" height="336" width="336" src="" />
180
+ </foreignObject>
181
+ </svg>
182
+ ```
183
+
184
+
16
185
 
17
186
 
18
187
  ## License
data/Rakefile CHANGED
@@ -6,7 +6,7 @@ Hoe.spec 'artq' do
6
6
 
7
7
  self.version = ArtQ::VERSION
8
8
 
9
- self.summary = "artq - query (ethereum) blockchain contracts / services for data about art collections via json-rpc"
9
+ self.summary = "artq - query (ethereum) blockchain contracts / services for (meta) data about art collections via json-rpc"
10
10
  self.description = summary
11
11
 
12
12
  self.urls = { home: 'https://github.com/pixelartexchange/artbase' }
data/bin/artq ADDED
@@ -0,0 +1,17 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ ###################
4
+ # == DEV TIPS:
5
+ #
6
+ # For local testing run like:
7
+ #
8
+ # ruby -Ilib bin/artq
9
+ #
10
+ # Set the executable bit in Linux. Example:
11
+ #
12
+ # % chmod a+x bin/artq
13
+ #
14
+
15
+ require 'artq'
16
+
17
+ ArtQ::Tool.main
data/lib/artq/version.rb CHANGED
@@ -1,7 +1,7 @@
1
1
 
2
2
 
3
3
  module ArtQ
4
- VERSION = '0.0.1'
4
+ VERSION = '0.1.0'
5
5
  end # module ArtQ
6
6
 
7
7
 
data/lib/artq.rb CHANGED
@@ -1,6 +1,136 @@
1
+ require 'ethlite'
2
+ require 'optparse'
1
3
 
2
4
 
3
- require_relative 'artq/version'
5
+ ## our own code
6
+ require_relative 'artq/version' # let version go first
7
+
8
+
9
+
10
+ module ArtQ
11
+
12
+ class Tool
13
+
14
+ def self.main( args=ARGV )
15
+ puts "==> welcome to artq tool with args:"
16
+ pp args
17
+
18
+ options = {
19
+ }
20
+
21
+ parser = OptionParser.new do |opts|
22
+
23
+ opts.on("--rpc STRING",
24
+ "JSON RPC Host (default: nil)") do |str|
25
+ options[ :rpc] = str
26
+ end
27
+
28
+ opts.on("-h", "--help", "Prints this help") do
29
+ puts opts
30
+ exit
31
+ end
32
+ end
33
+
34
+ parser.parse!( args )
35
+ puts "options:"
36
+ pp options
37
+
38
+ puts "args:"
39
+ pp args
40
+
41
+ if args.size < 1
42
+ puts "!! ERROR - no collection found - use <collection> <command>..."
43
+ puts ""
44
+ exit
45
+ end
46
+
47
+ contract_address = args[0] ## todo/check - use collection_name/slug or such?
48
+ command = args[1] || 'info'
49
+
50
+
51
+ if ['i','inf','info'].include?( command )
52
+ do_info( contract_address )
53
+ else
54
+ puts "!! ERROR - unknown command >#{command}<, sorry"
55
+ end
56
+
57
+ puts "bye"
58
+ end
59
+
60
+
61
+ ETH_name = Ethlite::ContractMethod.new( 'name',
62
+ inputs: [],
63
+ outputs: ['string'] )
64
+
65
+ ETH_symbol = Ethlite::ContractMethod.new( 'symbol',
66
+ inputs: [],
67
+ outputs: ['string'] )
68
+
69
+ ETH_totalSupply = Ethlite::ContractMethod.new( 'totalSupply',
70
+ inputs: [],
71
+ outputs: ['uint256'] )
72
+
73
+ ETH_tokenURI = Ethlite::ContractMethod.new( 'tokenURI',
74
+ inputs: ['uint256'],
75
+ outputs: ['string'] )
76
+
77
+
78
+
79
+ def self.do_info( contract_address )
80
+ puts "==> query art collection contract info @ >#{contract_address}<:"
81
+
82
+ rpc = JsonRpc.new( ENV['INFURA_URI'] )
83
+
84
+ name = ETH_name.do_call( rpc, contract_address, [] )
85
+ symbol = ETH_symbol.do_call( rpc, contract_address, [] )
86
+ totalSupply = ETH_totalSupply.do_call( rpc, contract_address, [] )
87
+
88
+ meta = []
89
+ tokenIds = (0..2)
90
+ tokenIds.each do |tokenId|
91
+ tokenURI = ETH_tokenURI.do_call( rpc, contract_address, [tokenId] )
92
+ meta << [tokenId, tokenURI]
93
+ end
94
+
95
+ puts " name: >#{name}<"
96
+ puts " symbol: >#{symbol}<"
97
+ puts " totalSupply: >#{totalSupply}<"
98
+ puts
99
+ puts " tokenURIs #{tokenIds}:"
100
+ meta.each do |tokenId, tokenURI|
101
+ puts " tokenId #{tokenId}:"
102
+ if tokenURI.start_with?( 'data:application/json;base64')
103
+ ## on-blockchain!!!
104
+ ## decode base64
105
+
106
+ str = tokenURI.sub( 'data:application/json;base64', '' )
107
+ str = Base64.decode64( str )
108
+ data = JSON.parse( str )
109
+
110
+
111
+ ## check for image_data - and replace if base64 encoded
112
+ image_data = data['image_data']
113
+
114
+ if image_data.start_with?( 'data:image/svg+xml;base64' )
115
+ data['image_data'] = '...'
116
+ str = image_data.sub( 'data:image/svg+xml;base64', '' )
117
+ image_data = Base64.decode64( str )
118
+ end
119
+
120
+ pp data
121
+ puts
122
+ puts " image_data:"
123
+ puts image_data
124
+ else
125
+ puts " #{tokenURI}"
126
+ end
127
+ end
128
+ end
129
+ end # class Tool
130
+
131
+
132
+ end # module ArtQ
133
+
4
134
 
5
135
 
6
136
 
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: artq
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.0.1
4
+ version: 0.1.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Gerald Bauer
@@ -58,10 +58,11 @@ dependencies:
58
58
  - - "~>"
59
59
  - !ruby/object:Gem::Version
60
60
  version: '3.23'
61
- description: artq - query (ethereum) blockchain contracts / services for data about
62
- art collections via json-rpc
61
+ description: artq - query (ethereum) blockchain contracts / services for (meta) data
62
+ about art collections via json-rpc
63
63
  email: wwwmake@googlegroups.com
64
- executables: []
64
+ executables:
65
+ - artq
65
66
  extensions: []
66
67
  extra_rdoc_files:
67
68
  - CHANGELOG.md
@@ -72,6 +73,7 @@ files:
72
73
  - Manifest.txt
73
74
  - README.md
74
75
  - Rakefile
76
+ - bin/artq
75
77
  - lib/artq.rb
76
78
  - lib/artq/version.rb
77
79
  homepage: https://github.com/pixelartexchange/artbase
@@ -98,6 +100,6 @@ requirements: []
98
100
  rubygems_version: 3.3.7
99
101
  signing_key:
100
102
  specification_version: 4
101
- summary: artq - query (ethereum) blockchain contracts / services for data about art
102
- collections via json-rpc
103
+ summary: artq - query (ethereum) blockchain contracts / services for (meta) data about
104
+ art collections via json-rpc
103
105
  test_files: []