Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/misterhat/livelook
soulseek client in javascript
https://github.com/misterhat/livelook
javascript music nodejs p2p soulseek
Last synced: 2 months ago
JSON representation
soulseek client in javascript
- Host: GitHub
- URL: https://github.com/misterhat/livelook
- Owner: misterhat
- License: agpl-3.0
- Created: 2019-04-28T14:02:47.000Z (over 5 years ago)
- Default Branch: master
- Last Pushed: 2019-05-08T19:21:04.000Z (over 5 years ago)
- Last Synced: 2023-08-05T04:04:58.072Z (over 1 year ago)
- Topics: javascript, music, nodejs, p2p, soulseek
- Language: JavaScript
- Homepage:
- Size: 224 KB
- Stars: 83
- Watchers: 9
- Forks: 8
- Open Issues: 8
-
Metadata Files:
- Readme: README.md
- License: COPYING
Awesome Lists containing this project
README
# livelook 👀
a [soulseek](https://en.wikipedia.org/wiki/Soulseek) client written in
javascript. soulseek allows users from around the world to connect to each other
directly and share music (and other stuff).>Soulseek is an ad-free, spyware free, just plain free file sharing network for
>Windows, Mac and Linux. Our rooms, search engine and search correlation system
>make it easy for you to find people with similar interests, and make new
>discoveries!
>
>-- [About Soulseek | Soulseek](https://www.slsknet.org/news/node/680)features supported:
* [nat pmp](https://en.wikipedia.org/wiki/NAT_Port_Mapping_Protocol) and
[upnp](https://en.wikipedia.org/wiki/Universal_Plug_and_Play) port-forwarding
* chat rooms and private messages (including direct connections)
* browsing user's share lists and vice versa
* searching the network's files and responding to distributed searches
* downloading and uploading with automatically updating share list
i mainly tested this against nicotine-plus, but it works with soulseekqt too.## example
```javascript
const LiveLook = require('./');
const fs = require('fs');let livelook = new LiveLook({
username: 'toadtripler',
password: 'not my password',
sharedFolder: './mp3s',
autojoin: [ 'nicotine' ]
});livelook.on('error', console.error);
livelook.login((err, res) => {
if (err || !res.success) {
return console.log('login failed');
}livelook.on('sayChatroom', msg => {
console.log(`[${msg.room}] <${msg.username}> ${msg.message}`);
});livelook.on('messageUser', msg => {
console.log(`<${msg.username}> ${msg.message}`);
});livelook.search('hot dad web love', (err, res) => {
if (err) {
return console.error(err);
}res = res.filter(item => item.slotsFree);
if (!res) {
console.log('no files found :(');
return;
}let downloaded = fs.createWriteStream('hot-dad-web-love.mp3');
res = res.sort((a, b) => a.speed > b.speed ? -1: 0)[0];
livelook.downloadFile(res.username, res.file).pipe(downloaded);
downloaded.on('end', () => {
livelook.messageUser(res.username, `hey thanks for ${res.file}!`);
});
});
});
```
## install$ npm install --save livelook
## api
### livelook = new LiveLook(args)
create a new `livelook` instance.```javascript
// args
{
username: '',
password: '',
server: 'server.slsknet.org',
port: 2242, // port for server above, NOT the port we listen on
waitPort: 2234, // port for peer server. will retry multiple options if fail
sharedFolder: './mp3s',
downloadFolder: './downloads',
description: 'user biography',
autojoin: [ 'chatrooms', 'joined', 'automatically' ],
maxPeers: 100,
uploadSlots: 2, // maximum uploads allowed at one time
uploadThrottle: 56 * 1024, // speed to throttle uploads in bytes
downloadThrottle: 56 * 1024
}
```### livelook.init([done])
initialize our share list and connect to the soulseek server. you don't need to
call this if you just use login below.### livelook.login([username, password, done])
login to the soulseek server, and initialize our peer server if it isn't
already. username and password are optional if the instance has them.### livelook.refreshShareList([done])
re-scan `livelook.sharedFolder` and repopulate `livelook.shareList`. this is
what other users see when they browse us, or when we respond to searches.### livelook.sayChatroom(room, message)
send a message to a chatroom.### livelook.leaveChatroom(room)
leave a chatroom and stop receiving messages from it.### livelook.joinRoom(room)
join a chatroom and start accepting messages from it.### livelook.messageUser(username, message)
send a private message to a specific user.### livelook.setStatus(status)
set our online/away status.`status` can be a `Number` (1 for away, 2 for online), `'away'` or `'online'`.
### livelook.refreshUploadSpeed([done])
re-calculate our upload speed from [speedtest.net](https://www.speedtest.net/).### livelook.getPeerAddress(username, done)
get a peer's ip address and port based on their username.### livelook.getPeerByUsername(username, done)
get a peer instance based on a username. this will first check our pre-existing
peers, then it tries to make a direct connection to the peer until finally
requesting the server connect the peer to us.### livelook.getShareFileList(username, done)
get all the files a user is sharing. may take a while as some people share
large amounts of files, and the message must be decompressed.### livelook.searchUserShares(username, query, done)
search a user's shares for a query. nicotine users max out at 50 by default.### livelook.getUserInfo(username, done)
get a user's description (biography), picture (avatar) as a buffer,
upload slots, queue size and slots free.### livelook.getFolderContents(username, folder, done)
get a list of all the files in the specified folder.### livelook.downloadFile(username, file, [fileStart = 0])
download a file from a user. this returns a `ReadableStream`, and will also
emit a `queue` event with its position if we can't download it immediately. pass
in `fileStart` to indicate where to begin downloading the file in bytes (to
resume interrupted downloads).### livelook.searchFiles(query, [args = { timeout, max }, done])
search other peers for files. this returns a `ReadableStream` in `objectMode`.
on search results it will emit `data` events containing the following:```javascript
{ username, file, size, bitrate, vbr, duration, slotsFree, speed, queueSize }
```if `done` is provided, the terms will be concatted and passed in either after
the timeout finishes or the max results are reached.## how it works
soulseek is largely peer-to-peer, but still relies on a central server for
chat rooms, messaging, piercing firewalls and finding peers.### initializing
first we set up a local server to accept peer connections (and port forward with
nat pmp or upnp if possible). then we connect to slsknet.org (or any other
soulseek server) as a separate client and login. we can now begin to chat and
browse.### finding peers
we can connect to peers by fetching their ip and port based on their username
from the soulseek server, but if they aren't port-forwarded this will fail. the
next step is to send a connection request via the soulseek server to tell them
to connect to us. if this fails, there is no way for them to connect to us. this
is why it's a good idea to enable nat pmp or port forward manually.### searching
after logging in, we tell the server we're orphaned and have no parents. after
an arbitrary amount of time (usually around 30 seconds), the server gives a list
of potential parents. once we connect to one successfully, we can also become a
parent. our parent will contionously send us search requests from other peers.
we can respond to them directly if we have the results, but also send the
request to all of our children so they can do the same.## see also
* [Soulseek.NET](https://github.com/jpdillingham/Soulseek.NET) by @jpdillingham
* [museek-plus](https://github.com/eLvErDe/museek-plus) by @eLvErDe
* [protocol docs](https://htmlpreview.github.io/?http://github.com/misterhat/livelook/blob/master/doc/SoulseekProtocol%20%E2%80%93%20Museek%2B.html)
* [nicotine-plus](https://github.com/Nicotine-Plus/nicotine-plus)
* [slsk-client](https://github.com/f-hj/slsk-client) by @f-hj.
* [soleseek protocol docs](https://htmlpreview.github.io/?https://github.com/misterhat/livelook/blob/master/doc/soulseek_protocol.html)## donate
[donate to keep the central server alive!](https://www.slsknet.org/donate.php)## license
Copyright (C) 2019 Zorian MedwidThis program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.You should have received a copy of the GNU Affero General Public License
along with this program. If not, see http://www.gnu.org/licenses/.**You may not distribute this program without offering the source code. Hosting
a web service that utilizes livelook is distrubtion.**