dasch-bencode 0.5.0

Sign up to get free protection for your applications and to get access to all the features.
data/LICENSE ADDED
@@ -0,0 +1,20 @@
1
+
2
+ Copyright (c) 2009 Daniel Schierbeck
3
+
4
+ Permission is hereby granted, free of charge, to any person obtaining a copy
5
+ of this software and associated documentation files (the "Software"), to deal
6
+ in the Software without restriction, including without limitation the rights
7
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8
+ copies of the Software, and to permit persons to whom the Software is
9
+ furnished to do so, subject to the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be included in
12
+ all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
19
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
20
+ THE SOFTWARE.
@@ -0,0 +1,40 @@
1
+
2
+ Ruby bencode binding
3
+ ====================
4
+
5
+ This is a simple library for reading and writing bencoded data.
6
+
7
+
8
+ What is bencode?
9
+ ----------------
10
+
11
+ Bencode is a simple data serialization format used by the popular
12
+ [BitTorrent](http://bittorrent.org/) P2P file sharing system.
13
+
14
+ It contains only four data types, namely:
15
+
16
+ - byte strings
17
+ - integers
18
+ - lists
19
+ - dictionaries
20
+
21
+
22
+ Examples
23
+ --------
24
+
25
+ "foo bar".bencode # => "7:foo bar"
26
+ 42.bencode # => "i42e"
27
+ [1, 2, 3].bencode # => "li1ei2ei3ee"
28
+ {"foo" => 1, "bar" => -10}.bencode # => "d3:bari-10e3:fooi1ee"
29
+
30
+
31
+ License
32
+ -------
33
+
34
+ Released under the MIT license.
35
+
36
+
37
+ Contributors
38
+ ------------
39
+
40
+ - Mike Hodgson
@@ -0,0 +1,14 @@
1
+
2
+ require 'rake/testtask'
3
+ require 'rake/rdoctask'
4
+
5
+ ##
6
+ # Run all unit tests.
7
+ Rake::TestTask.new
8
+
9
+
10
+ ##
11
+ # Generate RDoc documentation.
12
+ Rake::RDocTask.new :doc do |rdoc|
13
+ rdoc.rdoc_dir = 'doc'
14
+ end
@@ -0,0 +1,9 @@
1
+
2
+ # TODO: Write some documentation here.
3
+ module BEncode
4
+ VERSION = '0.5.0'
5
+ end
6
+
7
+ glob = File.join(File.dirname(__FILE__), 'bencode/**/*.rb')
8
+
9
+ Dir[glob].sort.each{|file| require file }
@@ -0,0 +1,83 @@
1
+
2
+ module BEncode
3
+ class << self
4
+ # BEncodes +obj+
5
+ def dump(obj)
6
+ obj.bencode
7
+ end
8
+
9
+ # Bdecodes +str+
10
+ def load(str)
11
+ require 'strscan'
12
+
13
+ scanner = StringScanner.new(str)
14
+ obj = parse(scanner)
15
+ raise BEncode::DecodeError unless scanner.eos?
16
+ return obj
17
+ end
18
+
19
+ # Bdecodes the file located at +path+
20
+ def load_file(path)
21
+ if RUBY_PLATFORM =~ /(win|w)32$/
22
+ load(File.open(path, 'rb').read)
23
+ else
24
+ load(File.open(path).read)
25
+ end
26
+ end
27
+
28
+ def parse(scanner) # :nodoc:
29
+ case scanner.scan(/[ild]|\d+:/)
30
+ when "i"
31
+ number = scanner.scan(/0|(?:-?[1-9][0-9]*)/)
32
+ raise BEncode::DecodeError unless number and scanner.scan(/e/)
33
+ return number.to_i
34
+ when "l"
35
+ ary = []
36
+ # TODO: There must be a smarter way of doing this...
37
+ ary.push(parse(scanner)) until scanner.peek(1) == "e"
38
+ scanner.pos += 1
39
+ return ary
40
+ when "d"
41
+ hsh = {}
42
+ until scanner.peek(1) == "e"
43
+ key, value = parse(scanner), parse(scanner)
44
+
45
+ unless key.is_a? String
46
+ raise BEncode::DecodeError, "key must be a string"
47
+ end
48
+
49
+ hsh.store(key, value)
50
+ end
51
+ scanner.pos += 1
52
+ return hsh
53
+ when /\d+:/
54
+ length = Integer($~.string.chop)
55
+ str = scanner.peek(length)
56
+
57
+ begin
58
+ scanner.pos += length
59
+ rescue RangeError
60
+ raise BEncode::DecodeError, "invalid string length"
61
+ end
62
+
63
+ return str
64
+ else
65
+ raise BEncode::DecodeError
66
+ end
67
+ end
68
+
69
+ private :parse
70
+ end
71
+ end
72
+
73
+ class String
74
+ #
75
+ # Bdecodes the String object and returns the data serialized
76
+ # through bencoding.
77
+ #
78
+ # "li1ei2ei3ee".bdecode #=> [1, 2, 3]
79
+ #
80
+ def bdecode
81
+ BEncode.load(self)
82
+ end
83
+ end
@@ -0,0 +1,4 @@
1
+
2
+ unless BEncode.const_defined? :DecodeError
3
+ BEncode::DecodeError = Class.new(StandardError)
4
+ end
@@ -0,0 +1,16 @@
1
+
2
+ class Array
3
+ #
4
+ # Bencodes the Array object. Bencoded arrays are represented as
5
+ # +lxe+, where +x+ is zero or more bencoded objects.
6
+ #
7
+ # [1, "foo"].bencode #=> "li1e3:fooe"
8
+ #
9
+ def bencode
10
+ begin
11
+ "l#{map{|obj| obj.bencode }.join('')}e"
12
+ rescue BEncode::EncodeError
13
+ raise BEncode::EncodeError, "array items must be encodable"
14
+ end
15
+ end
16
+ end
@@ -0,0 +1,15 @@
1
+
2
+ class Hash
3
+ #
4
+ # Bencodes the Hash object. Bencoded hashes are represented as
5
+ # +dxe+, where +x+ is zero or a power of two bencoded objects.
6
+ # each key is immediately followed by its associated value.
7
+ # All keys must be strings. The keys of the bencoded hash will
8
+ # be in lexicographical order.
9
+ def bencode
10
+ pairs = sort.map{|key, val| [key.to_str.bencode, val.bencode] }
11
+ "d#{pairs.join('')}e"
12
+ rescue NoMethodError => error
13
+ raise BEncode::EncodeError, "dictionary keys must be strings"
14
+ end
15
+ end
@@ -0,0 +1,13 @@
1
+
2
+ class Integer
3
+ #
4
+ # Bencodes the Integer object. Bencoded integers are represented
5
+ # as +ixe+, where +x+ is the integer with an optional
6
+ # hyphen prepended, indicating negativity.
7
+ #
8
+ # 42.bencode #=> "i42e"
9
+ # -7.bencode #=> "i-7e"
10
+ def bencode
11
+ "i#{self}e"
12
+ end
13
+ end
@@ -0,0 +1,9 @@
1
+
2
+ class Object
3
+ #
4
+ # Raises an exception. Subclasses of Object must themselves
5
+ # define meaningful #bencode methods.
6
+ def bencode
7
+ raise BEncode::EncodeError, "cannot bencode #{self.class}"
8
+ end
9
+ end
@@ -0,0 +1,14 @@
1
+
2
+ class String
3
+ #
4
+ # Bencodes the String object. Bencoded strings are represented
5
+ # as <code>x</code>:<code>y</code>, where +y+ is the string and +x+
6
+ # is the length of the string.
7
+ #
8
+ # "foo".bencode #=> "3:foo"
9
+ # "".bencode #=> "0:"
10
+ #
11
+ def bencode
12
+ "#{length}:#{self}"
13
+ end
14
+ end
@@ -0,0 +1,4 @@
1
+
2
+ unless BEncode.const_defined? :EncodeError
3
+ BEncode::EncodeError = Class.new(StandardError)
4
+ end
@@ -0,0 +1,18 @@
1
+
2
+ class IO
3
+ def self.bdecode(filename)
4
+ open(filename, 'r').bdecode
5
+ end
6
+
7
+ def self.bencode(filename)
8
+ open(filename, 'r').bencode
9
+ end
10
+
11
+ def bdecode
12
+ read.chomp.bdecode
13
+ end
14
+
15
+ def bencode
16
+ read.chomp.bencode
17
+ end
18
+ end
@@ -0,0 +1,2 @@
1
+
2
+ require File.dirname(__FILE__) + '/../lib/bencode'
@@ -0,0 +1,13 @@
1
+
2
+ require 'test/unit'
3
+ require 'test/environment'
4
+
5
+ class TestDecodeUTF8 < Test::Unit::TestCase
6
+
7
+ def test_can_handle_utf8_encoded_data
8
+ assert_nothing_raised do
9
+ BEncode.load_file('test/fixtures/test.torrent')
10
+ end
11
+ end
12
+
13
+ end
metadata ADDED
@@ -0,0 +1,67 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: dasch-bencode
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.5.0
5
+ platform: ruby
6
+ authors:
7
+ - Daniel Schierbeck
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+
12
+ date: 2009-05-16 00:00:00 -07:00
13
+ default_executable:
14
+ dependencies: []
15
+
16
+ description: Ruby bindings for the bencode data serialization format.
17
+ email: daniel.schierbeck@gmail.com
18
+ executables: []
19
+
20
+ extensions: []
21
+
22
+ extra_rdoc_files: []
23
+
24
+ files:
25
+ - Rakefile
26
+ - LICENSE
27
+ - README.md
28
+ - lib/bencode.rb
29
+ - lib/bencode/decode.rb
30
+ - lib/bencode/decode_error.rb
31
+ - lib/bencode/encode_error.rb
32
+ - lib/bencode/io.rb
33
+ - lib/bencode/encode/array.rb
34
+ - lib/bencode/encode/integer.rb
35
+ - lib/bencode/encode/string.rb
36
+ - lib/bencode/encode/hash.rb
37
+ - lib/bencode/encode/object.rb
38
+ has_rdoc: true
39
+ homepage: http://github.com/dasch/ruby-bencode-bindings
40
+ post_install_message:
41
+ rdoc_options: []
42
+
43
+ require_paths:
44
+ - lib
45
+ required_ruby_version: !ruby/object:Gem::Requirement
46
+ requirements:
47
+ - - ">="
48
+ - !ruby/object:Gem::Version
49
+ version: "0"
50
+ version:
51
+ required_rubygems_version: !ruby/object:Gem::Requirement
52
+ requirements:
53
+ - - ">="
54
+ - !ruby/object:Gem::Version
55
+ version: "0"
56
+ version:
57
+ requirements: []
58
+
59
+ rubyforge_project:
60
+ rubygems_version: 1.2.0
61
+ signing_key:
62
+ specification_version: 2
63
+ summary: Ruby bencode bindings
64
+ test_files:
65
+ - test/test_decode_utf8.rb
66
+ - test/environment.rb
67
+ - test/fixtures/test.torrent