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 +4 -4
- data/CHANGELOG.md +8 -0
- data/README.md +77 -5
- data/lib/blue_factory/configuration.rb +1 -1
- data/lib/blue_factory/errors.rb +12 -0
- data/lib/blue_factory/server.rb +65 -21
- data/lib/blue_factory/tasks/publish.rake +13 -0
- data/lib/blue_factory/version.rb +1 -1
- metadata +3 -6
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: de68a9c6f0b159edc06d950cb3ad558537273ce95637a3c2b2ab183f0eeb52d9
|
4
|
+
data.tar.gz: 609d5127c0ee34fbc19bca7b3cb37b01cbde365e05f24699e8cd4b0bd226d340
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
|
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
|
-
|
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 ©
|
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
|
data/lib/blue_factory/errors.rb
CHANGED
@@ -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
|
data/lib/blue_factory/server.rb
CHANGED
@@ -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
|
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
|
-
|
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
|
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
|
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
|
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,
|
data/lib/blue_factory/version.rb
CHANGED
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.
|
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:
|
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.
|
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: []
|