theoldreader 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
data/.gitignore ADDED
@@ -0,0 +1,17 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ Gemfile.lock
7
+ InstalledFiles
8
+ _yardoc
9
+ coverage
10
+ doc/
11
+ lib/bundler/man
12
+ pkg
13
+ rdoc
14
+ spec/reports
15
+ test/tmp
16
+ test/version_tmp
17
+ tmp
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in theoldreader.gemspec
4
+ gemspec
data/LICENSE.txt ADDED
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2013 Ian
2
+
3
+ MIT License
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ "Software"), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be
14
+ included in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,55 @@
1
+ # Theoldreader
2
+
3
+ TODO: Write a gem description
4
+
5
+ ## Installation
6
+
7
+ Add this line to your application's Gemfile:
8
+
9
+ gem 'theoldreader'
10
+
11
+ And then execute:
12
+
13
+ $ bundle
14
+
15
+ Or install it yourself as:
16
+
17
+ $ gem install theoldreader
18
+
19
+ ## Usage
20
+
21
+ Set token
22
+
23
+ Theoldreader.set_token('token')
24
+
25
+ Get the server status
26
+
27
+ Theoldreader.status
28
+
29
+ Get the token
30
+
31
+ Theoldreader.token
32
+
33
+ Get user info
34
+
35
+ Theoldreader.user_info
36
+
37
+ Get preference
38
+
39
+ Theoldreader.preference
40
+
41
+ Get friends list
42
+
43
+ Theoldreader.friends
44
+
45
+ Get tags
46
+
47
+ Theoldreader.tags
48
+
49
+ ## Contributing
50
+
51
+ 1. Fork it
52
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
53
+ 3. Commit your changes (`git commit -am 'Add some feature'`)
54
+ 4. Push to the branch (`git push origin my-new-feature`)
55
+ 5. Create new Pull Request
data/Rakefile ADDED
@@ -0,0 +1 @@
1
+ require "bundler/gem_tasks"
@@ -0,0 +1,116 @@
1
+ require 'multi_json'
2
+
3
+ require 'theoldreader/version'
4
+ require 'theoldreader/api/service'
5
+ require 'theoldreader/errors'
6
+ require 'theoldreader/core_ext/string'
7
+ require 'theoldreader/core_ext/hash'
8
+ require 'theoldreader/base'
9
+ require 'theoldreader/config'
10
+
11
+ module Theoldreader
12
+
13
+ def self.set_token(token)
14
+ Theoldreader::Config.token = token
15
+ end
16
+
17
+ def self.status
18
+ Theoldreader::Base.get({}, endpoint: 'status')
19
+ end
20
+
21
+ def self.token
22
+ Theoldreader::Base.get({}, endpoint: 'token')
23
+ end
24
+
25
+ def self.user_info
26
+ Theoldreader::Base.get({}, endpoint: 'user-info')
27
+ end
28
+
29
+ def self.preference
30
+ Theoldreader::Base.get({}, endpoint: 'preference/list')
31
+ end
32
+
33
+ def self.friends
34
+ Theoldreader::Base.get({}, endpoint: 'friend/list')
35
+ end
36
+
37
+ def self.tags
38
+ Theoldreader::Base.get({}, endpoint: 'tag/list')
39
+ end
40
+
41
+ def self.stream_preference
42
+ Theoldreader::Base.get({}, endpoint: 'preference/stream/list')
43
+ end
44
+
45
+ def self.update_stream_preference(params)
46
+ Theoldreader::Base.post({}, endpoint: 'preference/stream/set')
47
+ end
48
+
49
+ def self.rename_tag(src, dest)
50
+ Theoldreader::Base.post({s: src, dest: dest}, endpoint: 'rename-tag')
51
+ end
52
+
53
+ def self.delete_tag(tag_name)
54
+ Theoldreader::Base.post({s: tag_name}, endpoint: 'disable-tag')
55
+ end
56
+
57
+ def self.unread_count
58
+ Theoldreader::Base.get({}, endpoint: 'unread-count')
59
+ end
60
+
61
+ def self.subscriptions
62
+ Theoldreader::Base.get({}, endpoint: 'subscription/list')
63
+ end
64
+
65
+ def self.subscription_opml
66
+ Theoldreader::API::Service.instance.make_request(:get, 'subscriptions/export', {}, Theoldreader::Base.auth_header, {base_path: '/reader'}).body
67
+ end
68
+
69
+ def self.add_subscription(subscription)
70
+ Theoldreader::Base.post({quickadd: subscription}, endpoint: 'subscription/quickadd')
71
+ end
72
+
73
+ def self.update_subscription(subscription_id, params = {})
74
+ filtered_params = filter_params(%w{t a r}, params)
75
+ Theoldreader::Base.post(filtered_params.merge({ac: 'edit', s: subscription_id}), endpoint: 'subscription/edit')
76
+ end
77
+
78
+ def self.delete_subscription(subscription_id)
79
+ Theoldreader::Base.post({ac: 'unsubscribe', s: subscription_id}, endpoint: 'subscription/edit')
80
+ end
81
+
82
+ def self.item_ids(params = {})
83
+ filtered_params = filter_params(%w{s xt n r c nt ot}, params)
84
+ Theoldreader::Base.get(filtered_params, endpoint: 'stream/items/ids')
85
+ end
86
+
87
+ def self.items(params = {})
88
+ filtered_params = filter_params(%w{i output}, params)
89
+ Theoldreader::Base.post(filtered_params, endpoint: 'stream/items/contents')
90
+ end
91
+
92
+ def self.stream(params = {})
93
+ filtered_params = filter_params(%w{i output}, params)
94
+ Theoldreader::Base.get(filtered_params, endpoint: 'stream/contents')
95
+ end
96
+
97
+ def self.mark_all_as_read(params = {})
98
+ filtered_params = filter_params(%w{s ts}, params)
99
+ Theoldreader::Base.post(filtered_params, endpoint: 'mark-all-as-read')
100
+ end
101
+
102
+ def self.update_items(params = {})
103
+ filtered_params = filter_params(%w{i output}, params)
104
+ Theoldreader::Base.post(filtered_params, endpoint: 'edit-tag')
105
+ end
106
+
107
+ def self.atom_feed(path)
108
+ Theoldreader::API::Service.instance.make_request(:get, path, {}, Theoldreader::Base.auth_header, {base_path: '/reader/atom'}).body
109
+ end
110
+
111
+ private
112
+
113
+ def self.filter_params(fields, params)
114
+ params.slice(*fields.concat(fields.map(&:to_sym)))
115
+ end
116
+ end
@@ -0,0 +1,13 @@
1
+ module Theoldreader
2
+ module API
3
+ class Response
4
+ attr_reader :status, :body, :headers
5
+
6
+ def initialize(status, body, headers)
7
+ @status = status
8
+ @body = body
9
+ @headers = headers
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,61 @@
1
+ require 'faraday'
2
+ require 'singleton'
3
+ require 'theoldreader/api/response'
4
+
5
+ module Theoldreader
6
+ module API
7
+
8
+ class Service
9
+ include Singleton
10
+
11
+ HOST = 'theoldreader.com'
12
+ BASE_PATH = '/reader/api/0'
13
+ DEFAULT_MIDDLEWARE = Proc.new do |builder|
14
+ builder.request :url_encoded
15
+ builder.adapter Faraday.default_adapter
16
+ end
17
+ FARADAY_OPTIONS = %w{request proxy ssl builder url parallel_manager params headers builder_class}.map(&:to_sym)
18
+
19
+ attr_accessor :faraday_middleware
20
+ # config of the http options for all requests from the client
21
+ # http_options can be set through Service.instance.http_options[:use_ssl] = false
22
+ attr_accessor :http_options
23
+
24
+ def initialize
25
+ @http_options = {}
26
+ end
27
+
28
+ def make_request(verb, path, params, headers, options = {})
29
+ # use ssl by default
30
+ options[:use_ssl] ||= true
31
+ options[:base_path] ||= BASE_PATH
32
+ options[:host] ||= HOST
33
+ # per request options will override the client options
34
+ request_options = http_options.merge(options)
35
+
36
+ conn = Faraday.new(service_url(options[:use_ssl], options[:host], options[:base_path]),
37
+ faraday_options(request_options), &(faraday_middleware || DEFAULT_MIDDLEWARE))
38
+ response = conn.send(verb, path, params, headers)
39
+ Theoldreader::API::Response.new(response.status.to_i, response.body, response.headers)
40
+ end
41
+
42
+ %w{get put post delete}.each do |verb|
43
+ define_method verb do |*args|
44
+ make_request(verb, *args)
45
+ end
46
+ end
47
+
48
+ private
49
+
50
+ def faraday_options(options)
51
+ Hash[ options.select { |key,value| FARADAY_OPTIONS.include?(key) } ]
52
+ end
53
+
54
+ def service_url(use_ssl = true, host = HOST, base_path = BASE_PATH)
55
+ protocol = use_ssl ? 'https' : 'http'
56
+ "#{protocol}://#{host + base_path}"
57
+ end
58
+ end
59
+
60
+ end
61
+ end
@@ -0,0 +1,66 @@
1
+ require 'theoldreader/api/service'
2
+ require 'theoldreader/errors'
3
+
4
+ module Theoldreader
5
+ class Base
6
+
7
+ %w{get post put delete}.each do |method|
8
+ define_singleton_method(method) do |params = {}, options = {}|
9
+ headers = options.delete(:headers) || {}
10
+ url = options.delete(:endpoint) || endpoint
11
+ response = service.send(method, url, {output: format}.merge(params),
12
+ auth_header.merge(headers), options)
13
+ handle_errors(response)
14
+ begin
15
+ MultiJson.load(response.body.to_s)
16
+ rescue MultiJson::LoadError
17
+ response.body
18
+ end
19
+ end
20
+ end
21
+
22
+ def self.auth_header
23
+ token.nil? ? {} : {'Authorization' => "GoogleLogin auth=#{token}"}
24
+ end
25
+
26
+ private
27
+
28
+ def self.handle_errors(response)
29
+ if response.status >= 400
30
+ begin
31
+ response_hash = MultiJson.load(response.body)
32
+ rescue MultiJson::DecodeError
33
+ response_hash = {}
34
+ end
35
+
36
+ error_info = {
37
+ 'code' => response.status,
38
+ 'message' => response_hash['errors']
39
+ }
40
+
41
+ if response.status >= 500
42
+ raise Theoldreader::ServerError.new(response.status, response.body, error_info)
43
+ else
44
+ raise Theoldreader::ClientError.new(response.status, response.body, error_info)
45
+ end
46
+ end
47
+ end
48
+
49
+ def self.endpoint
50
+ @endpoint ||= self.name.split('::').last.hyphenize
51
+ end
52
+
53
+ def self.service
54
+ @service ||= Theoldreader::API::Service.instance
55
+ end
56
+
57
+ def self.token
58
+ Theoldreader::Config.token
59
+ end
60
+
61
+ def self.format
62
+ 'json'
63
+ end
64
+
65
+ end
66
+ end
@@ -0,0 +1,18 @@
1
+ require 'yaml'
2
+
3
+ module Theoldreader
4
+ class Config
5
+ class << self
6
+ attr_accessor :token
7
+ def load
8
+ @config ||= YAML::load_file(File.join(File.dirname(File.expand_path(__FILE__)), 'theoldreader.yml'))
9
+ rescue Errno::ENOENT
10
+ {}
11
+ end
12
+
13
+ def token
14
+ @token ||= load['token']
15
+ end
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,6 @@
1
+ class Hash
2
+ def slice(*keys)
3
+ keys.map! { |key| convert_key(key) } if respond_to?(:convert_key, true)
4
+ keys.each_with_object(self.class.new) { |k, hash| hash[k] = self[k] if has_key?(k) }
5
+ end
6
+ end
@@ -0,0 +1,21 @@
1
+ class String
2
+ def hyphenize
3
+ gsub(/::/, '/').
4
+ gsub(/([A-Z]+)([A-Z][a-z])/,'\1_\2').
5
+ gsub(/([a-z\d])([A-Z])/,'\1_\2').
6
+ tr("_", "-").
7
+ tr(" ", "-").
8
+ downcase
9
+ end
10
+
11
+ def underscore(camel_cased_word)
12
+ word = camel_cased_word.to_s.dup
13
+ word.gsub!('::', '/')
14
+ word.gsub!(/(?:([A-Za-z\d])|^)(#{inflections.acronym_regex})(?=\b|[^a-z])/) { "#{$1}#{$1 && '_'}#{$2.downcase}" }
15
+ word.gsub!(/([A-Z\d]+)([A-Z][a-z])/,'\1_\2')
16
+ word.gsub!(/([a-z\d])([A-Z])/,'\1_\2')
17
+ word.tr!("-", "_")
18
+ word.downcase!
19
+ word
20
+ end
21
+ end
@@ -0,0 +1,15 @@
1
+ module Theoldreader
2
+ class TheoldreaderError < StandardError; end
3
+ class APIError < TheoldreaderError
4
+ attr_reader :status, :response_body, :errors
5
+
6
+ def initialize(status, response_body, errors)
7
+ super("status: #{status}, body: #{response_body}, errors: #{errors.to_s}")
8
+ end
9
+ end
10
+
11
+ # Any error with a 5xx HTTP status code
12
+ class ServerError < APIError; end
13
+ # Any error with a 4xx HTTP status code
14
+ class ClientError < APIError; end
15
+ end
@@ -0,0 +1,3 @@
1
+ module Theoldreader
2
+ VERSION = "0.0.1"
3
+ end
@@ -0,0 +1,25 @@
1
+ # -*- encoding: utf-8 -*-
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'theoldreader/version'
5
+
6
+ Gem::Specification.new do |gem|
7
+ gem.name = "theoldreader"
8
+ gem.version = Theoldreader::VERSION
9
+ gem.authors = ["Ian Zhang"]
10
+ gem.email = ["ian7zy@gmail.com"]
11
+ gem.description = %q{Ruby integrations for theoldreader API}
12
+ gem.summary = %q{Ruby integrations for theoldreader API}
13
+ gem.homepage = ""
14
+
15
+ gem.files = `git ls-files`.split($/)
16
+ gem.executables = gem.files.grep(%r{^bin/}).map{ |f| File.basename(f) }
17
+ gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
18
+ gem.require_paths = ["lib"]
19
+
20
+ gem.add_runtime_dependency("multi_json")
21
+ gem.add_runtime_dependency("faraday")
22
+ gem.add_runtime_dependency("addressable")
23
+ gem.add_development_dependency("rspec")
24
+ gem.add_development_dependency("rake")
25
+ end
metadata ADDED
@@ -0,0 +1,140 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: theoldreader
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ prerelease:
6
+ platform: ruby
7
+ authors:
8
+ - Ian Zhang
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+ date: 2013-12-22 00:00:00.000000000 Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
15
+ name: multi_json
16
+ requirement: !ruby/object:Gem::Requirement
17
+ none: false
18
+ requirements:
19
+ - - ! '>='
20
+ - !ruby/object:Gem::Version
21
+ version: '0'
22
+ type: :runtime
23
+ prerelease: false
24
+ version_requirements: !ruby/object:Gem::Requirement
25
+ none: false
26
+ requirements:
27
+ - - ! '>='
28
+ - !ruby/object:Gem::Version
29
+ version: '0'
30
+ - !ruby/object:Gem::Dependency
31
+ name: faraday
32
+ requirement: !ruby/object:Gem::Requirement
33
+ none: false
34
+ requirements:
35
+ - - ! '>='
36
+ - !ruby/object:Gem::Version
37
+ version: '0'
38
+ type: :runtime
39
+ prerelease: false
40
+ version_requirements: !ruby/object:Gem::Requirement
41
+ none: false
42
+ requirements:
43
+ - - ! '>='
44
+ - !ruby/object:Gem::Version
45
+ version: '0'
46
+ - !ruby/object:Gem::Dependency
47
+ name: addressable
48
+ requirement: !ruby/object:Gem::Requirement
49
+ none: false
50
+ requirements:
51
+ - - ! '>='
52
+ - !ruby/object:Gem::Version
53
+ version: '0'
54
+ type: :runtime
55
+ prerelease: false
56
+ version_requirements: !ruby/object:Gem::Requirement
57
+ none: false
58
+ requirements:
59
+ - - ! '>='
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ - !ruby/object:Gem::Dependency
63
+ name: rspec
64
+ requirement: !ruby/object:Gem::Requirement
65
+ none: false
66
+ requirements:
67
+ - - ! '>='
68
+ - !ruby/object:Gem::Version
69
+ version: '0'
70
+ type: :development
71
+ prerelease: false
72
+ version_requirements: !ruby/object:Gem::Requirement
73
+ none: false
74
+ requirements:
75
+ - - ! '>='
76
+ - !ruby/object:Gem::Version
77
+ version: '0'
78
+ - !ruby/object:Gem::Dependency
79
+ name: rake
80
+ requirement: !ruby/object:Gem::Requirement
81
+ none: false
82
+ requirements:
83
+ - - ! '>='
84
+ - !ruby/object:Gem::Version
85
+ version: '0'
86
+ type: :development
87
+ prerelease: false
88
+ version_requirements: !ruby/object:Gem::Requirement
89
+ none: false
90
+ requirements:
91
+ - - ! '>='
92
+ - !ruby/object:Gem::Version
93
+ version: '0'
94
+ description: Ruby integrations for theoldreader API
95
+ email:
96
+ - ian7zy@gmail.com
97
+ executables: []
98
+ extensions: []
99
+ extra_rdoc_files: []
100
+ files:
101
+ - .gitignore
102
+ - Gemfile
103
+ - LICENSE.txt
104
+ - README.md
105
+ - Rakefile
106
+ - lib/theoldreader.rb
107
+ - lib/theoldreader/api/response.rb
108
+ - lib/theoldreader/api/service.rb
109
+ - lib/theoldreader/base.rb
110
+ - lib/theoldreader/config.rb
111
+ - lib/theoldreader/core_ext/hash.rb
112
+ - lib/theoldreader/core_ext/string.rb
113
+ - lib/theoldreader/errors.rb
114
+ - lib/theoldreader/version.rb
115
+ - theoldreader.gemspec
116
+ homepage: ''
117
+ licenses: []
118
+ post_install_message:
119
+ rdoc_options: []
120
+ require_paths:
121
+ - lib
122
+ required_ruby_version: !ruby/object:Gem::Requirement
123
+ none: false
124
+ requirements:
125
+ - - ! '>='
126
+ - !ruby/object:Gem::Version
127
+ version: '0'
128
+ required_rubygems_version: !ruby/object:Gem::Requirement
129
+ none: false
130
+ requirements:
131
+ - - ! '>='
132
+ - !ruby/object:Gem::Version
133
+ version: '0'
134
+ requirements: []
135
+ rubyforge_project:
136
+ rubygems_version: 1.8.24
137
+ signing_key:
138
+ specification_version: 3
139
+ summary: Ruby integrations for theoldreader API
140
+ test_files: []