| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233 | <?phpnamespace App\Http\Controllers;use App\Models\SensitiveWords;use App\Models\SsNode;use App\Models\User;use Exception;use Illuminate\Foundation\Auth\Access\AuthorizesRequests;use Illuminate\Foundation\Bus\DispatchesJobs;use Illuminate\Foundation\Validation\ValidatesRequests;use Illuminate\Http\UploadedFile;use Illuminate\Routing\Controller as BaseController;use RuntimeException;use Str;class Controller extends BaseController {	use AuthorizesRequests, DispatchesJobs, ValidatesRequests;	// 生成随机密码	public function makePasswd() {		return makeRandStr();	}	// 生成UUID	public function makeUUID() {		return Str::uuid();	}	// 生成网站安全码	public function makeSecurityCode(): string {		return strtolower(makeRandStr(8));	}	// 类似Linux中的tail命令	public function tail($file, $n, $base = 5) {		$fileLines = $this->countLine($file);		if($fileLines < 15000){			return false;		}		$fp = fopen($file, 'rb+');		assert($n > 0);		$pos = $n + 1;		$lines = [];		while(count($lines) <= $n){			try{				fseek($fp, -$pos, SEEK_END);			}catch(Exception $e){				break;			}			$pos *= $base;			while(!feof($fp)){				array_unshift($lines, fgets($fp));			}		}		return array_slice($lines, 0, $n);	}	/**	 * 计算文件行数	 *	 * @param $file	 *	 * @return int	 */	public function countLine($file): int {		$fp = fopen($file, 'rb');		$i = 0;		while(!feof($fp)){			//每次读取2M			if($data = fread($fp, 1024 * 1024 * 2)){				//计算读取到的行数				$num = substr_count($data, "\n");				$i += $num;			}		}		fclose($fp);		return $i;	}	// 获取敏感词	public function sensitiveWords($type): array {		return SensitiveWords::query()->whereType($type)->get()->pluck('words')->toArray();	}	// 将Base64图片转换为本地图片并保存	public function base64ImageSaver($base64_image_content): ?string {		// 匹配出图片的格式		if(preg_match('/^(data:\s*image\/(\w+);base64,)/', $base64_image_content, $result)){			$type = $result[2];			$directory = date('Ymd');			$path = '/assets/images/qrcode/'.$directory.'/';			// 检查是否有该文件夹,如果没有就创建,并给予最高权限			if(!file_exists(public_path($path))			   && !mkdir($concurrentDirectory = public_path($path), 0755, true)			   && !is_dir($concurrentDirectory)){				throw new RuntimeException(sprintf('Directory "%s" was not created', $concurrentDirectory));			}			$fileName = makeRandStr(18, true).".{$type}";			if(file_put_contents(public_path($path.$fileName),				base64_decode(str_replace($result[1], '', $base64_image_content)))){				chmod(public_path($path.$fileName), 0744);				return $path.$fileName;			}		}		return '';	}	// 上传文件处理	public function uploadFile(UploadedFile $file): string {		$fileType = $file->getClientOriginalExtension();		// 验证文件合法性		if(!in_array($fileType, ['jpg', 'png', 'jpeg', 'bmp'])){			return false;		}		$name = date('YmdHis').random_int(1000, 2000).'.'.$fileType;		$move = $file->move(base_path().'/public/upload/image/', $name);		return $move? '/upload/image/'.$name : '';	}	/**	 * 节点信息	 *	 * @param  int  $uid       用户ID	 * @param  int  $nodeId    节点ID	 * @param  int  $infoType  信息类型:0为链接,1为文字	 *	 * @return string	 */	public function getUserNodeInfo($uid, $nodeId, $infoType): string {		$user = User::whereId($uid)->firstOrFail();		$node = SsNode::whereId($nodeId)->firstOrFail();		$scheme = null;		// 获取分组名称		$group = $node->getLevel->name;		$host = $node->server?: $node->ip;		$data = null;		switch($node->type){			case 2:				// 生成v2ray scheme				if($infoType !== 1){					// 生成v2ray scheme					$data = $this->v2raySubUrl($node->name, $host, $node->v2_port, $user->vmess_id, $node->v2_alter_id,						$node->v2_net, $node->v2_type, $node->v2_host, $node->v2_path, $node->v2_tls? "tls" : "");				}else{					$data = "服务器:".$host.PHP_EOL."IPv6:".($node->ipv6?: "").PHP_EOL."端口:".$node->v2_port.PHP_EOL."加密方式:".$node->v2_method.PHP_EOL."用户ID:".$user->vmess_id.PHP_EOL."额外ID:".$node->v2_alter_id.PHP_EOL."传输协议:".$node->v2_net.PHP_EOL."伪装类型:".$node->v2_type.PHP_EOL."伪装域名:".($node->v2_host?: "").PHP_EOL."路径:".($node->v2_path?: "").PHP_EOL."TLS:".($node->v2_tls? "tls" : "").PHP_EOL;				}				break;			case 3:				if($infoType !== 1){					$data = $this->trojanSubUrl($user->passwd, $host, $node->port, $node->name);				}else{					$data = "备注:".$node->name.PHP_EOL."服务器:".$host.PHP_EOL."密码:".$user->passwd.PHP_EOL."端口:".$node->port.PHP_EOL;				}				break;			case 1:			case 4:				$protocol = $node->protocol;				$method = $node->method;				$obfs = $node->obfs;				if($node->single){					$port = $node->port;					$passwd = $node->passwd;					$protocol_param = $user->port.':'.$user->passwd;				}else{					$port = $user->port;					$passwd = $user->passwd;					$protocol_param = $node->protocol_param;					if($node->type === 1){						$protocol = $user->protocol;						$method = $user->method;						$obfs = $user->obfs;					}				}				if($infoType !== 1){					// 生成ss/ssr scheme					$data = $node->compatible? $this->ssSubUrl($host, $port, $method, $passwd,						$group) : $this->ssrSubUrl($host, $port, $protocol, $method, $obfs, $passwd, $node->obfs_param,						$protocol_param, $node->name, $group, $node->is_udp);				}else{					// 生成文本配置信息					$data = "服务器:".$host.PHP_EOL."IPv6:".$node->ipv6.PHP_EOL."服务器端口:".$port.PHP_EOL."密码:".$passwd.PHP_EOL."加密:".$method.PHP_EOL.($node->compatible? '' : "协议:".$protocol.PHP_EOL."协议参数:".$protocol_param.PHP_EOL."混淆:".$obfs.PHP_EOL."混淆参数:".$node->obfs_param.PHP_EOL);				}				break;			default:		}		return $data;	}	public function v2raySubUrl($name, $host, $port, $uuid, $alter_id, $net, $type, $domain, $path, $tls): string {		return 'vmess://'.base64url_encode(json_encode([				"v"    => "2",				"ps"   => $name,				"add"  => $host,				"port" => $port,				"id"   => $uuid,				"aid"  => $alter_id,				"net"  => $net,				"type" => $type,				"host" => $domain,				"path" => $path,				"tls"  => $tls? "tls" : ""			], JSON_PRETTY_PRINT));	}	public function trojanSubUrl($password, $domain, $port, $remark): string {		return 'trojan://'.urlencode($password).'@'.$domain.':'.$port.'#'.urlencode($remark);	}	public function ssSubUrl($host, $port, $method, $passwd, $group): string {		return 'ss://'.base64url_encode($method.':'.$passwd.'@'.$host.':'.$port).'#'.$group;	}	public function ssrSubUrl(		$host, $port, $protocol, $method, $obfs, $passwd, $obfs_param, $protocol_param, $name, $group, $is_udp	): string {		return 'ssr://'.base64url_encode($host.':'.$port.':'.$protocol.':'.$method.':'.$obfs.':'.base64url_encode($passwd).'/?obfsparam='.base64url_encode($obfs_param).'&protoparam='.base64url_encode($protocol_param).'&remarks='.base64url_encode($name).'&group='.base64url_encode($group).'&udpport='.$is_udp.'&uot=0');	}}
 |