mirror of
https://git.bsd.gay/fef/nyastodon.git
synced 2024-12-24 18:13:42 +01:00
Merge commit 'f89512fbedb547f66a72eefdff047768fb505eb6' into glitch-soc/merge-upstream
Conflicts: - `README.md`: Upstream updated its README, we have a completely different one. Kept our version.
This commit is contained in:
commit
45e56db8e4
14 changed files with 598 additions and 440 deletions
2
.github/workflows/crowdin-download.yml
vendored
2
.github/workflows/crowdin-download.yml
vendored
|
@ -53,7 +53,7 @@ jobs:
|
||||||
|
|
||||||
# Create or update the pull request
|
# Create or update the pull request
|
||||||
- name: Create Pull Request
|
- name: Create Pull Request
|
||||||
uses: peter-evans/create-pull-request@v6.0.0
|
uses: peter-evans/create-pull-request@v6.0.1
|
||||||
with:
|
with:
|
||||||
commit-message: 'New Crowdin translations'
|
commit-message: 'New Crowdin translations'
|
||||||
title: 'New Crowdin Translations (automated)'
|
title: 'New Crowdin Translations (automated)'
|
||||||
|
|
|
@ -535,7 +535,7 @@ GEM
|
||||||
rack (2.2.8.1)
|
rack (2.2.8.1)
|
||||||
rack-attack (6.7.0)
|
rack-attack (6.7.0)
|
||||||
rack (>= 1.0, < 4)
|
rack (>= 1.0, < 4)
|
||||||
rack-cors (2.0.1)
|
rack-cors (2.0.2)
|
||||||
rack (>= 2.0.0)
|
rack (>= 2.0.0)
|
||||||
rack-oauth2 (1.21.3)
|
rack-oauth2 (1.21.3)
|
||||||
activesupport
|
activesupport
|
||||||
|
|
2
Vagrantfile
vendored
2
Vagrantfile
vendored
|
@ -188,7 +188,7 @@ Vagrant.configure(VAGRANTFILE_API_VERSION) do |config|
|
||||||
|
|
||||||
config.vm.post_up_message = <<MESSAGE
|
config.vm.post_up_message = <<MESSAGE
|
||||||
To start server
|
To start server
|
||||||
$ vagrant ssh -c "cd /vagrant && foreman start"
|
$ vagrant ssh -c "cd /vagrant && bin/dev"
|
||||||
MESSAGE
|
MESSAGE
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -127,7 +127,7 @@ module LanguagesHelper
|
||||||
om: ['Oromo', 'Afaan Oromoo'].freeze,
|
om: ['Oromo', 'Afaan Oromoo'].freeze,
|
||||||
or: ['Oriya', 'ଓଡ଼ିଆ'].freeze,
|
or: ['Oriya', 'ଓଡ଼ିଆ'].freeze,
|
||||||
os: ['Ossetian', 'ирон æвзаг'].freeze,
|
os: ['Ossetian', 'ирон æвзаг'].freeze,
|
||||||
pa: ['Panjabi', 'ਪੰਜਾਬੀ'].freeze,
|
pa: ['Punjabi', 'ਪੰਜਾਬੀ'].freeze,
|
||||||
pi: ['Pāli', 'पाऴि'].freeze,
|
pi: ['Pāli', 'पाऴि'].freeze,
|
||||||
pl: ['Polish', 'Polski'].freeze,
|
pl: ['Polish', 'Polski'].freeze,
|
||||||
ps: ['Pashto', 'پښتو'].freeze,
|
ps: ['Pashto', 'پښتو'].freeze,
|
||||||
|
@ -191,15 +191,18 @@ module LanguagesHelper
|
||||||
chr: ['Cherokee', 'ᏣᎳᎩ ᎦᏬᏂᎯᏍᏗ'].freeze,
|
chr: ['Cherokee', 'ᏣᎳᎩ ᎦᏬᏂᎯᏍᏗ'].freeze,
|
||||||
ckb: ['Sorani (Kurdish)', 'سۆرانی'].freeze,
|
ckb: ['Sorani (Kurdish)', 'سۆرانی'].freeze,
|
||||||
cnr: ['Montenegrin', 'crnogorski'].freeze,
|
cnr: ['Montenegrin', 'crnogorski'].freeze,
|
||||||
|
csb: ['Kashubian', 'Kaszëbsczi'].freeze,
|
||||||
jbo: ['Lojban', 'la .lojban.'].freeze,
|
jbo: ['Lojban', 'la .lojban.'].freeze,
|
||||||
kab: ['Kabyle', 'Taqbaylit'].freeze,
|
kab: ['Kabyle', 'Taqbaylit'].freeze,
|
||||||
ldn: ['Láadan', 'Láadan'].freeze,
|
ldn: ['Láadan', 'Láadan'].freeze,
|
||||||
lfn: ['Lingua Franca Nova', 'lingua franca nova'].freeze,
|
lfn: ['Lingua Franca Nova', 'lingua franca nova'].freeze,
|
||||||
|
pdc: ['Pennsylvania Dutch', 'Pennsilfaani-Deitsch'].freeze,
|
||||||
sco: ['Scots', 'Scots'].freeze,
|
sco: ['Scots', 'Scots'].freeze,
|
||||||
sma: ['Southern Sami', 'Åarjelsaemien Gïele'].freeze,
|
sma: ['Southern Sami', 'Åarjelsaemien Gïele'].freeze,
|
||||||
smj: ['Lule Sami', 'Julevsámegiella'].freeze,
|
smj: ['Lule Sami', 'Julevsámegiella'].freeze,
|
||||||
szl: ['Silesian', 'ślůnsko godka'].freeze,
|
szl: ['Silesian', 'ślůnsko godka'].freeze,
|
||||||
tok: ['Toki Pona', 'toki pona'].freeze,
|
tok: ['Toki Pona', 'toki pona'].freeze,
|
||||||
|
vai: ['Vai', 'ꕙꔤ'].freeze,
|
||||||
xal: ['Kalmyk', 'Хальмг келн'].freeze,
|
xal: ['Kalmyk', 'Хальмг келн'].freeze,
|
||||||
zba: ['Balaibalan', 'باليبلن'].freeze,
|
zba: ['Balaibalan', 'باليبلن'].freeze,
|
||||||
zgh: ['Standard Moroccan Tamazight', 'ⵜⴰⵎⴰⵣⵉⵖⵜ'].freeze,
|
zgh: ['Standard Moroccan Tamazight', 'ⵜⴰⵎⴰⵣⵉⵖⵜ'].freeze,
|
||||||
|
|
|
@ -2,31 +2,15 @@
|
||||||
|
|
||||||
class AccountSuggestions::FriendsOfFriendsSource < AccountSuggestions::Source
|
class AccountSuggestions::FriendsOfFriendsSource < AccountSuggestions::Source
|
||||||
def get(account, limit: DEFAULT_LIMIT)
|
def get(account, limit: DEFAULT_LIMIT)
|
||||||
Account.find_by_sql([<<~SQL.squish, { id: account.id, limit: limit }]).map { |row| [row.id, key] }
|
first_degree = account.following.where.not(hide_collections: true).select(:id).reorder(nil)
|
||||||
WITH first_degree AS (
|
base_account_scope(account)
|
||||||
SELECT target_account_id
|
.joins(:account_stat)
|
||||||
FROM follows
|
.where(id: Follow.where(account_id: first_degree).select(:target_account_id))
|
||||||
JOIN accounts AS target_accounts ON follows.target_account_id = target_accounts.id
|
.group('accounts.id, account_stats.id')
|
||||||
WHERE account_id = :id
|
.reorder('frequency DESC, followers_count DESC')
|
||||||
AND NOT target_accounts.hide_collections
|
.limit(limit)
|
||||||
)
|
.pluck(Arel.sql('accounts.id, COUNT(*) AS frequency'))
|
||||||
SELECT accounts.id, COUNT(*) AS frequency
|
.map { |id, _frequency| [id, key] }
|
||||||
FROM accounts
|
|
||||||
JOIN follows ON follows.target_account_id = accounts.id
|
|
||||||
JOIN account_stats ON account_stats.account_id = accounts.id
|
|
||||||
LEFT OUTER JOIN follow_recommendation_mutes ON follow_recommendation_mutes.target_account_id = accounts.id AND follow_recommendation_mutes.account_id = :id
|
|
||||||
WHERE follows.account_id IN (SELECT * FROM first_degree)
|
|
||||||
AND NOT EXISTS (SELECT 1 FROM follows f WHERE f.target_account_id = follows.target_account_id AND f.account_id = :id)
|
|
||||||
AND follows.target_account_id <> :id
|
|
||||||
AND accounts.discoverable
|
|
||||||
AND accounts.suspended_at IS NULL
|
|
||||||
AND accounts.silenced_at IS NULL
|
|
||||||
AND accounts.moved_to_account_id IS NULL
|
|
||||||
AND follow_recommendation_mutes.target_account_id IS NULL
|
|
||||||
GROUP BY accounts.id, account_stats.id
|
|
||||||
ORDER BY frequency DESC, account_stats.followers_count ASC
|
|
||||||
LIMIT :limit
|
|
||||||
SQL
|
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
|
@ -51,7 +51,8 @@ class AccountSuggestions::SimilarProfilesSource < AccountSuggestions::Source
|
||||||
recently_followed_account_ids = account.active_relationships.recent.limit(5).pluck(:target_account_id)
|
recently_followed_account_ids = account.active_relationships.recent.limit(5).pluck(:target_account_id)
|
||||||
|
|
||||||
if Chewy.enabled? && !recently_followed_account_ids.empty?
|
if Chewy.enabled? && !recently_followed_account_ids.empty?
|
||||||
QueryBuilder.new(recently_followed_account_ids, account).build.limit(limit).hits.pluck('_id').map(&:to_i).zip([key].cycle)
|
ids_from_es = QueryBuilder.new(recently_followed_account_ids, account).build.limit(limit).hits.pluck('_id').map(&:to_i)
|
||||||
|
base_account_scope(account).where(id: ids_from_es).pluck(:id).zip([key].cycle)
|
||||||
else
|
else
|
||||||
[]
|
[]
|
||||||
end
|
end
|
||||||
|
|
|
@ -12,6 +12,8 @@ class AccountSuggestions::Source
|
||||||
def base_account_scope(account)
|
def base_account_scope(account)
|
||||||
Account
|
Account
|
||||||
.searchable
|
.searchable
|
||||||
|
.where(discoverable: true)
|
||||||
|
.without_silenced
|
||||||
.where.not(follows_sql, id: account.id)
|
.where.not(follows_sql, id: account.id)
|
||||||
.where.not(follow_requests_sql, id: account.id)
|
.where.not(follow_requests_sql, id: account.id)
|
||||||
.not_excluded_by_account(account)
|
.not_excluded_by_account(account)
|
||||||
|
|
20
bin/dev
Executable file
20
bin/dev
Executable file
|
@ -0,0 +1,20 @@
|
||||||
|
#!/usr/bin/env sh
|
||||||
|
|
||||||
|
# Default to port 3000 if not specified
|
||||||
|
export PORT="${PORT:-3000}"
|
||||||
|
|
||||||
|
# Get around our boot.rb ENV check
|
||||||
|
export RAILS_ENV="${RAILS_ENV:-development}"
|
||||||
|
|
||||||
|
if command -v overmind &> /dev/null
|
||||||
|
then
|
||||||
|
overmind start -f Procfile.dev "$@"
|
||||||
|
exit $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
if gem list --no-installed --exact --silent foreman; then
|
||||||
|
echo "Installing foreman..."
|
||||||
|
gem install foreman
|
||||||
|
fi
|
||||||
|
|
||||||
|
foreman start -f Procfile.dev "$@"
|
|
@ -42,6 +42,7 @@ Rails.application.configure do
|
||||||
:hu,
|
:hu,
|
||||||
:hy,
|
:hy,
|
||||||
:id,
|
:id,
|
||||||
|
:ie,
|
||||||
:ig,
|
:ig,
|
||||||
:io,
|
:io,
|
||||||
:is,
|
:is,
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
{
|
{
|
||||||
"name": "@mastodon/mastodon",
|
"name": "@mastodon/mastodon",
|
||||||
"license": "AGPL-3.0-or-later",
|
"license": "AGPL-3.0-or-later",
|
||||||
"packageManager": "yarn@4.1.0",
|
"packageManager": "yarn@4.1.1",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18"
|
"node": ">=18"
|
||||||
},
|
},
|
||||||
|
|
|
@ -0,0 +1,82 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe AccountSuggestions::FriendsOfFriendsSource do
|
||||||
|
describe '#get' do
|
||||||
|
subject { described_class.new }
|
||||||
|
|
||||||
|
let!(:bob) { Fabricate(:account, discoverable: true, hide_collections: false) }
|
||||||
|
let!(:alice) { Fabricate(:account, discoverable: true, hide_collections: true) }
|
||||||
|
let!(:eve) { Fabricate(:account, discoverable: true, hide_collections: false) }
|
||||||
|
let!(:mallory) { Fabricate(:account, discoverable: false, hide_collections: false) }
|
||||||
|
let!(:eugen) { Fabricate(:account, discoverable: true, hide_collections: false) }
|
||||||
|
let!(:john) { Fabricate(:account, discoverable: true, hide_collections: false) }
|
||||||
|
let!(:jerk) { Fabricate(:account, discoverable: true, hide_collections: false) }
|
||||||
|
let!(:neil) { Fabricate(:account, discoverable: true, hide_collections: false) }
|
||||||
|
let!(:larry) { Fabricate(:account, discoverable: true, hide_collections: false) }
|
||||||
|
|
||||||
|
context 'with follows and blocks' do
|
||||||
|
before do
|
||||||
|
bob.block!(jerk)
|
||||||
|
FollowRecommendationMute.create!(account: bob, target_account: neil)
|
||||||
|
|
||||||
|
# bob follows eugen, alice and larry
|
||||||
|
[eugen, alice, larry].each { |account| bob.follow!(account) }
|
||||||
|
|
||||||
|
# alice follows eve and mallory
|
||||||
|
[john, mallory].each { |account| alice.follow!(account) }
|
||||||
|
|
||||||
|
# eugen follows eve, john, jerk, larry and neil
|
||||||
|
[eve, mallory, jerk, larry, neil].each { |account| eugen.follow!(account) }
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'returns eligible accounts', :aggregate_failures do
|
||||||
|
results = subject.get(bob)
|
||||||
|
|
||||||
|
# eve is returned through eugen
|
||||||
|
expect(results).to include([eve.id, :friends_of_friends])
|
||||||
|
|
||||||
|
# john is not reachable because alice hides who she follows
|
||||||
|
expect(results).to_not include([john.id, :friends_of_friends])
|
||||||
|
|
||||||
|
# mallory is not discoverable
|
||||||
|
expect(results).to_not include([mallory.id, :friends_of_friends])
|
||||||
|
|
||||||
|
# larry is not included because he's followed already
|
||||||
|
expect(results).to_not include([larry.id, :friends_of_friends])
|
||||||
|
|
||||||
|
# jerk is blocked
|
||||||
|
expect(results).to_not include([jerk.id, :friends_of_friends])
|
||||||
|
|
||||||
|
# the suggestion for neil has already been rejected
|
||||||
|
expect(results).to_not include([neil.id, :friends_of_friends])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'with deterministic order' do
|
||||||
|
before do
|
||||||
|
# bob follows eve and mallory
|
||||||
|
[eve, mallory].each { |account| bob.follow!(account) }
|
||||||
|
|
||||||
|
# eve follows eugen, john, and jerk
|
||||||
|
[jerk, eugen, john].each { |account| eve.follow!(account) }
|
||||||
|
|
||||||
|
# mallory follows eugen, john, and neil
|
||||||
|
[neil, eugen, john].each { |account| mallory.follow!(account) }
|
||||||
|
|
||||||
|
john.follow!(eugen)
|
||||||
|
john.follow!(neil)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'returns eligible accounts in the expected order' do
|
||||||
|
expect(subject.get(bob)).to eq [
|
||||||
|
[eugen.id, :friends_of_friends], # followed by 2 friends, 3 followers total
|
||||||
|
[john.id, :friends_of_friends], # followed by 2 friends, 2 followers total
|
||||||
|
[neil.id, :friends_of_friends], # followed by 1 friend, 2 followers total
|
||||||
|
[jerk.id, :friends_of_friends], # followed by 1 friend, 1 follower total
|
||||||
|
]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -11,14 +11,16 @@ RSpec.describe AccountSuggestions::Source do
|
||||||
end
|
end
|
||||||
|
|
||||||
context 'with follows and follow requests' do
|
context 'with follows and follow requests' do
|
||||||
let!(:account_domain_blocked_account) { Fabricate(:account, domain: 'blocked.host') }
|
let!(:account_domain_blocked_account) { Fabricate(:account, domain: 'blocked.host', discoverable: true) }
|
||||||
let!(:account) { Fabricate(:account) }
|
let!(:account) { Fabricate(:account, discoverable: true) }
|
||||||
let!(:blocked_account) { Fabricate(:account) }
|
let!(:blocked_account) { Fabricate(:account, discoverable: true) }
|
||||||
let!(:eligible_account) { Fabricate(:account) }
|
let!(:eligible_account) { Fabricate(:account, discoverable: true) }
|
||||||
let!(:follow_recommendation_muted_account) { Fabricate(:account) }
|
let!(:follow_recommendation_muted_account) { Fabricate(:account, discoverable: true) }
|
||||||
let!(:follow_requested_account) { Fabricate(:account) }
|
let!(:follow_requested_account) { Fabricate(:account, discoverable: true) }
|
||||||
let!(:following_account) { Fabricate(:account) }
|
let!(:following_account) { Fabricate(:account, discoverable: true) }
|
||||||
let!(:moved_account) { Fabricate(:account, moved_to_account: Fabricate(:account)) }
|
let!(:moved_account) { Fabricate(:account, moved_to_account: Fabricate(:account), discoverable: true) }
|
||||||
|
let!(:silenced_account) { Fabricate(:account, silenced: true, discoverable: true) }
|
||||||
|
let!(:undiscoverable_account) { Fabricate(:account, discoverable: false) }
|
||||||
|
|
||||||
before do
|
before do
|
||||||
Fabricate :account_domain_block, account: account, domain: account_domain_blocked_account.domain
|
Fabricate :account_domain_block, account: account, domain: account_domain_blocked_account.domain
|
||||||
|
@ -40,6 +42,8 @@ RSpec.describe AccountSuggestions::Source do
|
||||||
.and not_include(follow_requested_account)
|
.and not_include(follow_requested_account)
|
||||||
.and not_include(following_account)
|
.and not_include(following_account)
|
||||||
.and not_include(moved_account)
|
.and not_include(moved_account)
|
||||||
|
.and not_include(silenced_account)
|
||||||
|
.and not_include(undiscoverable_account)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
{
|
{
|
||||||
"name": "@mastodon/streaming",
|
"name": "@mastodon/streaming",
|
||||||
"license": "AGPL-3.0-or-later",
|
"license": "AGPL-3.0-or-later",
|
||||||
"packageManager": "yarn@4.1.0",
|
"packageManager": "yarn@4.1.1",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18"
|
"node": ">=18"
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in a new issue