blue_factory 0.1.3 → 0.1.5

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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: c1a078159ccaad626c030e20a9ad18e1a408cc8897c27cddb9773c1bd1a6295c
4
- data.tar.gz: b7c45df10d3419f21745b760b586fe21e1eb262e9a12c72a5282e3ca0616eace
3
+ metadata.gz: de68a9c6f0b159edc06d950cb3ad558537273ce95637a3c2b2ab183f0eeb52d9
4
+ data.tar.gz: 609d5127c0ee34fbc19bca7b3cb37b01cbde365e05f24699e8cd4b0bd226d340
5
5
  SHA512:
6
- metadata.gz: 3347004aff37324480e13d2c1a21b85f737a9ab8898d874ecf26dad6f1b8f74ffa621ca24d4d76098be1479e145c08745d8c86a5cfcab68e66c8148bd6f76a8c
7
- data.tar.gz: 15fd3de8b85ecb2175c3d3ede61b73257bac08e849935ad42aad6b428d70c1fbe1fbec970887db354a6fea8f36fa93851f2909452497cc33f50419bff4cd1bea
6
+ metadata.gz: 25dee3b63c24fc48fce4b4aaa04ac8113c92e35df6464b8357053ef422fba14181c2cd3d7e26894fcbb14e736b029b2e5a3f45940d93fc294ddcbc52519f49bf
7
+ data.tar.gz: bad39da56a7a4cec3217dbf507197dc23c2951e855d8501bb89babe8a0966e3809065cbf9a0e1c7c038c77e0d5764a04feb8e6815c931962d40cd1bb29d5c391
data/CHANGELOG.md CHANGED
@@ -1,3 +1,11 @@
1
+ ## [0.1.5] - 2025-03-20
2
+
3
+ - added support for feed content mode field (video feeds)
4
+
5
+ ## [0.1.4] - 2023-08-19
6
+
7
+ - implemented partial authentication, without signature verification (`enable_unsafe_auth` option)
8
+
1
9
  ## [0.1.3] - 2023-07-27
2
10
 
3
11
  - fixed incorrect response when reaching the end of the feed
data/README.md CHANGED
@@ -1,6 +1,9 @@
1
1
  # BlueFactory 🏭
2
2
 
3
- A Ruby gem for hosting custom feeds for Bluesky
3
+ A Ruby gem for hosting custom feeds for Bluesky.
4
+
5
+ > [!NOTE]
6
+ > ATProto Ruby gems collection: [skyfall](https://github.com/mackuba/skyfall) | [blue_factory](https://github.com/mackuba/blue_factory) | [minisky](https://github.com/mackuba/minisky) | [didkit](https://github.com/mackuba/didkit)
4
7
 
5
8
 
6
9
  ## What does it do
@@ -40,7 +43,8 @@ BlueFactory.add_feed 'starwars', StarWarsFeed.new
40
43
 
41
44
  The `get_posts` method of the feed object should:
42
45
 
43
- - accept a single `params` argument which is a hash with fields: `:feed`, `:cursor` and `:limit` (the last two are optional)
46
+ - accept a `params` argument which is a hash with fields: `:feed`, `:cursor` and `:limit` (the last two are optional)
47
+ - optionally, accept a second `current_user` argument which is a string with the authenticated user's DID (depends on authentication config - [see below](#authentication))
44
48
  - return a hash with two fields: `:cursor` and `:posts`
45
49
 
46
50
  The `:feed` is the `at://` URI of the feed. The `:cursor` param, if included, should be a cursor returned by your feed from one of the previous requests, so it should be in the format used by the same function - but anyone can call the endpoint with any params, so you should validate it. The cursor is used for pagination to provide more pages further down in the feed (the first request to load the top of the feed doesn't include a cursor).
@@ -59,7 +63,7 @@ An example implementation could look like this:
59
63
  require 'time'
60
64
 
61
65
  class StarWarsFeed
62
- def get_posts(params)
66
+ def get_posts(params, current_user = nil)
63
67
  limit = check_query_limit(params)
64
68
  query = Post.select('uri, time').order('time DESC').limit(limit)
65
69
 
@@ -122,7 +126,71 @@ server {
122
126
  }
123
127
  ```
124
128
 
125
- ### Additional configuration & customizing
129
+ ## Authentication
130
+
131
+ Feeds are authenticated using a technology called [JSON Web Tokens](https://jwt.io). If a user is logged in, when they open, refresh or scroll down a feed in their app, requests are made to the feed service from the Bluesky network's IP address with user's authentication token in the `Authorization` HTTP header. (This is not the same kind of token as the access token that you use to make API calls - it does not let you perform any actions on user's behalf.)
132
+
133
+ At the moment, Blue Factory handles authentication in a very simplified way - it extracts the user's DID from the authentication header, but it does not verify the signature. This means that anyone with some programming knowledge can trivially prepare a fake token and make requests to the `getFeedSkeleton` endpoint as a different user.
134
+
135
+ As such, this authentication should not be used for anything critical. It may be used for things like logging, analytics, or as "security by obscurity" to just discourage others from accessing the feed in the app. You can also use this to build personalized feeds, as long as it's not a problem that the user DID may be fake.
136
+
137
+ To use this simple authentication, set the `enable_unsafe_auth` option:
138
+
139
+ ```rb
140
+ BlueFactory.set :enable_unsafe_auth, true
141
+ ```
142
+
143
+ The user's DID extracted from the token is passed as a second argument to `#get_posts`. You may, for example, return an empty list when the user is not authorized to use it:
144
+
145
+ ```rb
146
+ class HiddenFeed
147
+ def get_posts(params, current_user)
148
+ if AUTHORIZED_USERS.include?(current_user)
149
+ # ...
150
+ else
151
+ { posts: [] }
152
+ end
153
+ end
154
+ end
155
+ ```
156
+
157
+ Alternatively, you can raise a `BlueFactory::AuthorizationError` with an optional custom message. This will return a 401 status response to the Bluesky app, which will make it display the pink error banner in the app:
158
+
159
+ ```rb
160
+ class HiddenFeed
161
+ def get_posts(params, current_user)
162
+ if AUTHORIZED_USERS.include?(current_user)
163
+ # ...
164
+ else
165
+ raise BlueFactory::AuthorizationError, "You shall not pass!"
166
+ end
167
+ end
168
+ end
169
+ ```
170
+
171
+ <p><img width="400" src="https://github.com/mackuba/blue_factory/assets/28465/9197c0ec-9302-4ca0-b06c-3fce2e0fa4f4"></p>
172
+
173
+
174
+ ### Unauthenticated access
175
+
176
+ Please note that the `current_user` may be nil - this will happen if the authentication header is not set at all. Since the [bsky.app](https://bsky.app) website is now open to the public and can be accessed without authentication, people can also access your feeds without being logged in.
177
+
178
+ If you want the feed to only be available to logged in users (even if it's a non-personalized feed), simply raise an `AuthorizationError` if `current_user` is nil:
179
+
180
+ ```rb
181
+ class RestrictedFeed
182
+ def get_posts(params, current_user)
183
+ if current_user.nil?
184
+ raise BlueFactory::AuthorizationError, "Log in to see this feed"
185
+ end
186
+
187
+ # ...
188
+ end
189
+ end
190
+ ```
191
+
192
+
193
+ ## Additional configuration & customizing
126
194
 
127
195
  You can use the [Sinatra API](https://sinatrarb.com/intro.html#configuration) to do any additional configuration, like changing the server port, enabling/disabling logging and so on.
128
196
 
@@ -156,6 +224,7 @@ To publish the feed, you will need to provide some additional info about the fee
156
224
  - `display_name` (required) - the publicly visible name of your feed, e.g. "WWDC 23" (should be something short)
157
225
  - `description` (optional) - a longer (~1-2 lines) description of what the feed does, displayed on the feed page as the "bio"
158
226
  - `avatar_file` (optional) - path to an avatar image from the project's root (PNG or JPG)
227
+ - `content_mode` (optional) - return `:video` to create a video feed
159
228
 
160
229
  When you're ready, run the rake task passing the feed key (you will be asked for the uploader account's password):
161
230
 
@@ -163,9 +232,12 @@ When you're ready, run the rake task passing the feed key (you will be asked for
163
232
  bundle exec rake bluesky:publish KEY=wwdc
164
233
  ```
165
234
 
235
+ For non-Bluesky PDSes, you need to also add an env var `SERVER_URL=https://your.pds.host`.
236
+
237
+
166
238
  ## Credits
167
239
 
168
- Copyright © 2023 Kuba Suder ([@mackuba.eu](https://bsky.app/profile/mackuba.eu)).
240
+ Copyright © 2025 Kuba Suder ([@mackuba.eu](https://bsky.app/profile/mackuba.eu)).
169
241
 
170
242
  The code is available under the terms of the [zlib license](https://choosealicense.com/licenses/zlib/) (permissive, similar to MIT).
171
243
 
@@ -13,7 +13,7 @@ module BlueFactory
13
13
  (ENV['APP_ENV'] || ENV['RACK_ENV'] || :development).to_sym
14
14
  end
15
15
 
16
- configurable :publisher_did, :hostname, :validate_responses
16
+ configurable :publisher_did, :hostname, :validate_responses, :enable_unsafe_auth
17
17
 
18
18
  set :validate_responses, (environment != :production)
19
19
  end
@@ -1,4 +1,13 @@
1
1
  module BlueFactory
2
+ class AuthorizationError < StandardError
3
+ attr_reader :error_type
4
+
5
+ def initialize(message = "Authentication required", error_type = nil)
6
+ super(message)
7
+ @error_type = error_type
8
+ end
9
+ end
10
+
2
11
  class InvalidKeyError < StandardError
3
12
  end
4
13
 
@@ -13,4 +22,7 @@ module BlueFactory
13
22
 
14
23
  class InvalidResponseError < StandardError
15
24
  end
25
+
26
+ class UnsupportedAlgorithmError < StandardError
27
+ end
16
28
  end
@@ -1,3 +1,4 @@
1
+ require 'base64'
1
2
  require 'json'
2
3
  require 'sinatra/base'
3
4
 
@@ -12,7 +13,6 @@ module BlueFactory
12
13
  disable :static
13
14
  enable :quiet
14
15
  enable :logging
15
- set :default_content_type, 'application/json'
16
16
  settings.add_charset << 'application/json'
17
17
  end
18
18
 
@@ -25,14 +25,60 @@ module BlueFactory
25
25
  'at://' + config.publisher_did + '/' + FEED_GENERATOR_TYPE + '/' + key
26
26
  end
27
27
 
28
- def json(data)
28
+ def json_response(data)
29
+ content_type :json
29
30
  JSON.generate(data)
30
31
  end
31
32
 
33
+ alias json json_response
34
+
32
35
  def json_error(name, message, status: 400)
36
+ content_type :json
33
37
  [status, JSON.generate({ error: name, message: message })]
34
38
  end
35
39
 
40
+ def get_feed
41
+ if params[:feed].to_s.empty?
42
+ raise InvalidResponseError, "Error: Params must have the property \"feed\""
43
+ end
44
+
45
+ if params[:feed] !~ %r(^at://[\w\-\.\:]+/[\w\.]+/[\w\.\-]+$)
46
+ raise InvalidResponseError, "Error: feed must be a valid at-uri"
47
+ end
48
+
49
+ feed_key = params[:feed].split('/').last
50
+ feed = config.get_feed(feed_key)
51
+
52
+ if feed.nil? || feed_uri(feed_key) != params[:feed]
53
+ raise UnsupportedAlgorithmError, "Unsupported algorithm"
54
+ end
55
+
56
+ feed
57
+ end
58
+
59
+ def parse_did_from_token
60
+ auth = env['HTTP_AUTHORIZATION']
61
+
62
+ if auth.to_s.strip.empty?
63
+ return nil
64
+ end
65
+
66
+ if !auth.start_with?('Bearer ')
67
+ raise AuthorizationError, "Unsupported authorization method"
68
+ end
69
+
70
+ token = auth.gsub(/^Bearer /, '')
71
+ parts = token.split('.')
72
+ raise AuthorizationError.new("Invalid JWT format", "BadJwt") unless parts.length == 3
73
+
74
+ begin
75
+ payload = JSON.parse(Base64.decode64(parts[1]))
76
+ payload['iss']
77
+ rescue StandardError => e
78
+ raise AuthorizationError.new("Invalid JWT format", "BadJwt")
79
+ end
80
+ end
81
+
36
82
  def validate_response(response)
37
83
  cursor = response[:cursor]
38
84
  raise InvalidResponseError, ":cursor key is missing" unless response.has_key?(:cursor)
@@ -50,46 +96,44 @@ module BlueFactory
50
96
  end
51
97
 
52
98
  get '/xrpc/app.bsky.feed.getFeedSkeleton' do
53
- if params[:feed].to_s.empty?
54
- return json_error("InvalidRequest", "Error: Params must have the property \"feed\"")
55
- end
56
-
57
- if params[:feed] !~ %r(^at://[\w\-\.\:]+/[\w\.]+/[\w\.\-]+$)
58
- return json_error("InvalidRequest", "Error: feed must be a valid at-uri")
59
- end
60
-
61
- feed_key = params[:feed].split('/').last
62
- feed = config.get_feed(feed_key)
63
-
64
- if feed.nil? || feed_uri(feed_key) != params[:feed]
65
- return json_error("UnsupportedAlgorithm", "Unsupported algorithm")
66
- end
67
-
68
99
  begin
69
- response = feed.get_posts(params.slice(:feed, :cursor, :limit))
100
+ feed = get_feed
101
+ args = params.slice(:feed, :cursor, :limit)
102
+
103
+ if config.enable_unsafe_auth
104
+ did = parse_did_from_token
105
+ response = feed.get_posts(args, did)
106
+ else
107
+ response = feed.get_posts(args)
108
+ end
109
+
70
110
  validate_response(response) if config.validate_responses
71
111
 
72
112
  output = {}
73
113
  output[:feed] = response[:posts].map { |s| { post: s }}
74
114
  output[:cursor] = response[:cursor] if response[:cursor]
75
115
 
76
- return json(output)
116
+ return json_response(output)
77
117
  rescue InvalidRequestError => e
78
118
  return json_error(e.error_type || "InvalidRequest", e.message)
119
+ rescue AuthorizationError => e
120
+ return json_error(e.error_type || "AuthenticationRequired", e.message, status: 401)
121
+ rescue UnsupportedAlgorithmError => e
122
+ return json_error("UnsupportedAlgorithm", e.message)
79
123
  rescue InvalidResponseError => e
80
124
  return json_error("InvalidResponse", e.message)
81
125
  end
82
126
  end
83
127
 
84
128
  get '/xrpc/app.bsky.feed.describeFeedGenerator' do
85
- return json({
129
+ return json_response({
86
130
  did: config.service_did,
87
131
  feeds: config.feed_keys.map { |f| { uri: feed_uri(f) }}
88
132
  })
89
133
  end
90
134
 
91
135
  get '/.well-known/did.json' do
92
- return json({
136
+ return json_response({
93
137
  '@context': ['https://www.w3.org/ns/did/v1'],
94
138
  id: config.service_did,
95
139
  service: [
@@ -40,6 +40,18 @@ namespace :bluesky do
40
40
  feed_description = feed.description
41
41
  end
42
42
 
43
+ if feed.respond_to?(:content_mode)
44
+ case feed.content_mode
45
+ when nil, :unspecified
46
+ feed_content_mode = "app.bsky.feed.defs#contentModeUnspecified"
47
+ when :video
48
+ feed_content_mode = "app.bsky.feed.defs#contentModeVideo"
49
+ else
50
+ puts "Invalid content mode: #{feed.content_mode.inspect}. Accepted values: :video, :unspecified, nil."
51
+ exit 1
52
+ end
53
+ end
54
+
43
55
  if feed.respond_to?(:avatar_file) && feed.avatar_file.to_s.strip != ''
44
56
  avatar_file = feed.avatar_file
45
57
 
@@ -87,6 +99,7 @@ namespace :bluesky do
87
99
  }
88
100
 
89
101
  record[:avatar] = avatar_ref if avatar_ref
102
+ record[:contentMode] = feed_content_mode if feed_content_mode
90
103
 
91
104
  json = BlueFactory::Net.post_request(server, 'com.atproto.repo.putRecord', {
92
105
  repo: BlueFactory.publisher_did,
@@ -1,3 +1,3 @@
1
1
  module BlueFactory
2
- VERSION = "0.1.3"
2
+ VERSION = "0.1.5"
3
3
  end
metadata CHANGED
@@ -1,14 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: blue_factory
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.3
4
+ version: 0.1.5
5
5
  platform: ruby
6
6
  authors:
7
7
  - Kuba Suder
8
- autorequire:
9
8
  bindir: bin
10
9
  cert_chain: []
11
- date: 2023-07-27 00:00:00.000000000 Z
10
+ date: 2025-03-20 00:00:00.000000000 Z
12
11
  dependencies:
13
12
  - !ruby/object:Gem::Dependency
14
13
  name: sinatra
@@ -56,7 +55,6 @@ metadata:
56
55
  bug_tracker_uri: https://github.com/mackuba/blue_factory/issues
57
56
  changelog_uri: https://github.com/mackuba/blue_factory/blob/master/CHANGELOG.md
58
57
  source_code_uri: https://github.com/mackuba/blue_factory
59
- post_install_message:
60
58
  rdoc_options: []
61
59
  require_paths:
62
60
  - lib
@@ -71,8 +69,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
71
69
  - !ruby/object:Gem::Version
72
70
  version: '0'
73
71
  requirements: []
74
- rubygems_version: 3.4.10
75
- signing_key:
72
+ rubygems_version: 3.6.5
76
73
  specification_version: 4
77
74
  summary: A Ruby gem for hosting custom feeds for Bluesky
78
75
  test_files: []