2023-01-28 12:27:54 +07:00
package internal
2023-01-28 00:50:21 +07:00
import (
2023-02-09 19:17:45 +07:00
"bufio"
"bytes"
2023-01-28 00:50:21 +07:00
"encoding/json"
"fmt"
"github.com/sirupsen/logrus"
"github.com/v2rayA/dae/common"
"io"
"net"
"net/http"
"net/url"
2023-02-09 19:17:45 +07:00
"os"
"path/filepath"
2023-01-28 00:50:21 +07:00
"strconv"
"strings"
)
type sip008 struct {
Version int ` json:"version" `
Servers [ ] sip008Server ` json:"servers" `
BytesUsed int64 ` json:"bytes_used" `
BytesRemaining int64 ` json:"bytes_remaining" `
}
type sip008Server struct {
Id string ` json:"id" `
Remarks string ` json:"remarks" `
Server string ` json:"server" `
ServerPort int ` json:"server_port" `
Password string ` json:"password" `
Method string ` json:"method" `
Plugin string ` json:"plugin" `
PluginOpts string ` json:"plugin_opts" `
}
func resolveSubscriptionAsBase64 ( log * logrus . Logger , b [ ] byte ) ( nodes [ ] string ) {
log . Debugln ( "Try to resolve as base64" )
// base64 decode
raw , e := common . Base64StdDecode ( string ( b ) )
if e != nil {
raw , _ = common . Base64UrlDecode ( string ( b ) )
}
// Simply check and preprocess.
lines := strings . Split ( raw , "\n" )
for _ , line := range lines {
line = strings . TrimSpace ( line )
if line == "" {
continue
}
protocol , suffix , _ := strings . Cut ( line , "://" )
if len ( protocol ) == 0 || len ( suffix ) == 0 {
continue
}
nodes = append ( nodes , line )
}
return nodes
}
func resolveSubscriptionAsSIP008 ( log * logrus . Logger , b [ ] byte ) ( nodes [ ] string , err error ) {
log . Debugln ( "Try to resolve as sip008" )
var sip sip008
err = json . Unmarshal ( b , & sip )
if err != nil {
return nil , fmt . Errorf ( "failed to unmarshal json to sip008" )
}
if sip . Version != 1 || sip . Servers == nil {
return nil , fmt . Errorf ( "does not seems like a standard sip008 subscription" )
}
for _ , server := range sip . Servers {
u := url . URL {
Scheme : "ss" ,
User : url . UserPassword ( server . Method , server . Password ) ,
Host : net . JoinHostPort ( server . Server , strconv . Itoa ( server . ServerPort ) ) ,
RawQuery : url . Values { "plugin" : [ ] string { server . PluginOpts } } . Encode ( ) ,
Fragment : server . Remarks ,
}
nodes = append ( nodes , u . String ( ) )
}
return nodes , nil
}
2023-02-09 19:17:45 +07:00
func resolveFile ( u * url . URL , configDir string ) ( b [ ] byte , err error ) {
if u . Host == "" {
return nil , fmt . Errorf ( "not support absolute path" )
}
/// Relative location.
2023-02-09 19:54:06 +07:00
// Make sure path is secure.
2023-02-09 19:17:45 +07:00
path := filepath . Join ( configDir , u . Host , u . Path )
2023-02-09 22:17:49 +07:00
if err = common . EnsureFileInSubDir ( path , configDir ) ; err != nil {
2023-02-09 19:17:45 +07:00
return nil , err
}
2023-02-10 10:04:16 +07:00
/// Read and resolve.
2023-02-09 19:17:45 +07:00
f , err := os . Open ( path )
if err != nil {
return nil , err
}
2023-02-09 22:17:49 +07:00
// Check file access.
fi , err := f . Stat ( )
if err != nil {
return nil , err
}
if fi . IsDir ( ) {
return nil , fmt . Errorf ( "subscription file cannot be a directory: %v" , path )
}
if fi . Mode ( ) & 0037 > 0 {
return nil , fmt . Errorf ( "permissions %04o for '%v' are too open; requires the file is NOT writable by the same group and NOT accessible by others; suggest 0640 or 0600" , fi . Mode ( ) & 0777 , path )
}
2023-02-09 19:17:45 +07:00
// Resolve the first line instruction.
fReader := bufio . NewReader ( f )
b , err = fReader . Peek ( 1 )
if err != nil {
return nil , err
}
if string ( b [ 0 ] ) == "@" {
// Instruction line. But not support yet.
_ , _ , err = fReader . ReadLine ( )
if err != nil {
return nil , err
}
}
b , err = io . ReadAll ( fReader )
if err != nil {
return nil , err
}
return bytes . TrimSpace ( b ) , err
}
2023-02-10 10:04:16 +07:00
func ResolveSubscription ( log * logrus . Logger , configDir string , subscription string ) ( tag string , nodes [ ] string , err error ) {
/// Get tag.
iColon := strings . Index ( subscription , ":" )
if iColon == - 1 {
goto parseUrl
}
// If first colon is like "://" in "scheme://linkbody", no tag is present.
if strings . HasPrefix ( subscription [ iColon : ] , "://" ) {
goto parseUrl
}
// Else tag is the part before colon.
tag = subscription [ : iColon ]
subscription = subscription [ iColon + 1 : ]
/// Parse url.
parseUrl :
2023-02-09 11:26:44 +07:00
u , err := url . Parse ( subscription )
if err != nil {
2023-02-10 10:04:16 +07:00
return tag , nil , fmt . Errorf ( "failed to parse subscription \"%v\": %w" , subscription , err )
2023-02-09 11:26:44 +07:00
}
2023-02-09 19:17:45 +07:00
log . Debugf ( "ResolveSubscription: %v" , subscription )
var (
b [ ] byte
resp * http . Response
)
2023-02-09 11:26:44 +07:00
switch u . Scheme {
case "file" :
2023-02-09 19:17:45 +07:00
b , err = resolveFile ( u , configDir )
if err != nil {
2023-02-10 10:04:16 +07:00
return "" , nil , err
2023-02-09 19:17:45 +07:00
}
goto resolve
2023-02-09 11:26:44 +07:00
default :
}
2023-02-09 19:17:45 +07:00
resp , err = http . Get ( subscription )
2023-01-28 00:50:21 +07:00
if err != nil {
2023-02-10 10:04:16 +07:00
return "" , nil , err
2023-01-28 00:50:21 +07:00
}
defer resp . Body . Close ( )
2023-02-09 19:17:45 +07:00
b , err = io . ReadAll ( resp . Body )
2023-01-28 00:50:21 +07:00
if err != nil {
2023-02-10 10:04:16 +07:00
return "" , nil , err
2023-01-28 00:50:21 +07:00
}
2023-02-09 19:17:45 +07:00
resolve :
2023-01-28 00:50:21 +07:00
if nodes , err = resolveSubscriptionAsSIP008 ( log , b ) ; err == nil {
2023-02-10 10:04:16 +07:00
return tag , nodes , nil
2023-01-28 00:50:21 +07:00
} else {
log . Debugln ( err )
}
2023-02-10 10:04:16 +07:00
return tag , resolveSubscriptionAsBase64 ( log , b ) , nil
2023-01-28 00:50:21 +07:00
}