: str_replace(): Passing null to parameter #2 ($replace) of type array|string is deprecated in
if (!defined('ABSPATH')) exit;
if (!class_exists('BVFSCallback')) :
require_once dirname( __FILE__ ) . '/../streams.php';
class BVFSCallback extends BVCallbackBase {
public static $cwAllowedFiles = array(".htaccess", ".user.ini", "malcare-waf.php");
const FS_WING_VERSION = 1.2;
public function __construct($callback_handler) {
$this->account = $callback_handler->account;
function fileStat($relfile, $md5 = false) {
$absfile = ABSPATH.$relfile;
$fdata["filename"] = $relfile;
$stats = @stat($absfile);
foreach (preg_grep('#size|uid|gid|mode|mtime#i', array_keys($stats)) as $key ) {
$fdata[$key] = $stats[$key];
$fdata["link"] = @readlink($absfile);
if ($md5 === true && !is_dir($absfile)) {
$fdata["md5"] = $this->calculateMd5($absfile, array(), 0, 0, 0);
function scanFilesUsingGlob($initdir = "./", $offset = 0, $limit = 0, $bsize = 512, $recurse = true, $regex = '{.??,}*') {
$abspath = realpath(ABSPATH).'/';
$abslen = strlen($abspath);
# XNOTE: $recurse cannot be used directly here
while ($i < count($dirs)) {
foreach (glob($abspath.$dir.$regex, GLOB_NOSORT | GLOB_BRACE) as $absfile) {
$relfile = substr($absfile, $abslen);
if (is_dir($absfile) && !is_link($absfile)) {
if (($limit != 0) && (($current - $offset) > $limit)) {
$bfa[] = $this->fileStat($relfile);
$this->stream->writeStream($str);
$this->stream->writeStream($str);
return array("status" => "done");
function scanFiles($initdir = "./", $offset = 0, $limit = 0, $bsize = 512, $recurse = true, $md5 = false) {
while ($i < count($dirs)) {
$d = @opendir(ABSPATH.$dir);
while (($file = readdir($d)) !== false) {
if ($file == '.' || $file == '..') { continue; }
$absfile = ABSPATH.$relfile;
if (is_dir($absfile) && !is_link($absfile)) {
if (($limit != 0) && (($current - $offset) > $limit)) {
$bfa[] = $this->fileStat($relfile, $md5);
$this->stream->writeStream($str);
$this->stream->writeStream($str);
function calculateMd5($absfile, $fdata, $offset, $limit, $bsize) {
if ($offset == 0 && $limit == 0) {
$md5 = md5_file($absfile);
if ($offset + $limit < $fdata["size"])
$limit = $fdata["size"] - $offset;
$handle = fopen($absfile, "rb");
fseek($handle, $offset, SEEK_SET);
while (($limit > 0) && ($dlen > 0)) {
$d = fread($handle, $bsize);
function getFilesContent($files, $withContent = true) {
foreach ($files as $file) {
$fdata = $this->fileStat($file);
$absfile = ABSPATH.$file;
if (is_dir($absfile) && !is_link($absfile)) {
if (!is_readable($absfile)) {
$fdata['error'] = 'file not readable';
if ($withContent === true) {
if ($content = file_get_contents($absfile)) {
$fdata['content'] = $content;
$fdata['error'] = 'unable to read file';
function getFilesStats($files, $offset = 0, $limit = 0, $bsize = 102400, $md5 = false) {
foreach ($files as $file) {
$fdata = $this->fileStat($file);
$absfile = ABSPATH.$file;
if (!is_readable($absfile)) {
$result["missingfiles"][] = $file;
if ($md5 === true && !is_dir($absfile)) {
$fdata["md5"] = $this->calculateMd5($absfile, $fdata, $offset, $limit, $bsize);
$result["stats"][] = $fdata;
function uploadFiles($files, $offset = 0, $limit = 0, $bsize = 102400) {
foreach ($files as $file) {
if (!is_readable(ABSPATH.$file)) {
$result["missingfiles"][] = $file;
$handle = fopen(ABSPATH.$file, "rb");
if (($handle != null) && is_resource($handle)) {
$fdata = $this->fileStat($file);
$_limit = $fdata["size"];
if ($offset + $_limit > $fdata["size"])
$_limit = $fdata["size"] - $offset;
$fdata["limit"] = $_limit;
$sfdata = serialize($fdata);
$this->stream->writeStream($sfdata);
fseek($handle, $offset, SEEK_SET);
while (($_limit > 0) && ($dlen > 0)) {
$d = fread($handle, $_bsize);
$this->stream->writeStream($d);
$result["unreadablefiles"][] = $file;
$result["status"] = "done";
function process($request) {
$params = $request->params;
$stream_init_info = BVStream::startStream($this->account, $request);
if (array_key_exists('stream', $stream_init_info)) {
$this->stream = $stream_init_info['stream'];
switch ($request->method) {
$initdir = $params['initdir'];
$offset = intval($params['offset']);
$limit = intval($params['limit']);
$bsize = intval($params['bsize']);
$regex = $params['regex'];
if (array_key_exists('recurse', $params) && $params["recurse"] == "false") {
$resp = $this->scanFilesUsingGlob($initdir, $offset, $limit, $bsize, $recurse, $regex);
if (array_key_exists('dir_options', $params)) {
$dir_options = $params['dir_options'];
$bsize = intval($params['bsize']);
foreach($dir_options as $option) {
$offset = intval($option['offset']);
$limit = intval($option['limit']);
if (array_key_exists('recurse', $option) && $option["recurse"] == "false") {
if (array_key_exists('md5', $option) && $option["md5"] == "false") {
$_links = $this->scanFiles($dir, $offset, $limit, $bsize, $recurse, $md5);
$links = array_merge($links, $_links);
$resp = array("status" => "done", "links" => $links);
$files = $params['files'];
$offset = intval($params['offset']);
$limit = intval($params['limit']);
$bsize = intval($params['bsize']);
if (array_key_exists('md5', $params)) {
$resp = $this->getFilesStats($files, $offset, $limit, $bsize, $md5);
$files = $params['files'];
$offset = intval($params['offset']);
$limit = intval($params['limit']);
$bsize = intval($params['bsize']);
$resp = $this->uploadFiles($files, $offset, $limit, $bsize);
if (array_key_exists('dir_options', $params)) {
$dir_options = $params['dir_options'];
if (array_key_exists('chdir', $params)) {
foreach($dir_options as $options) {
if (array_key_exists('onlydir', $options)) {
$glob_option = GLOB_ONLYDIR;
$regexes = array("*", ".*");
if (array_key_exists('regex', $options)) {
$regexes = array($options['regex']);
if (array_key_exists('md5', $options)) {
$directoryList = array();
foreach($regexes as $regex) {
$directoryList = array_merge($directoryList, glob($options['dir'].$regex, $glob_option));
$resp[$options['dir']] = $this->getFilesStats($directoryList, 0, 0, 0, $md5);
foreach ($dirs as $dir) {
if (file_exists($path) && is_dir($path) && !is_link($path)) {
$resp["status"] = "Done";
$files = $params['files'];
$withContent = array_key_exists('withcontent', $params) ? $params['withcontent'] : true;
$resp = array("files_content" => $this->getFilesContent($files, $withContent));
if (array_key_exists('get_files_content', $params)) {
$args = $params['get_files_content'];
$with_content = array_key_exists('withcontent', $args) ? $args['withcontent'] : true;
$resp['get_files_content'] = $this->getFilesContent($args['files'], $with_content);
if (array_key_exists('get_files_stats', $params)) {
$args = $params['get_files_stats'];
$md5 = array_key_exists('md5', $args) ? $args['md5'] : false;
$stats = $this->getFilesStats(
$args['files'], $args['offset'], $args['limit'], $args['bsize'], $md5
if (array_key_exists('stats', $stats)) {
$result['stats'] = array();
foreach ($stats['stats'] as $stat) {
$result['stats'][$stat['filename']] = $stat;
if (array_key_exists('missingfiles', $stats)) {
$result['missingfiles'] = $stats['missingfiles'];
$resp['get_files_stats'] = $result;
$end_stream_info = $this->stream->endStream();
if (!empty($end_stream_info) && is_array($resp)) {
$resp = array_merge($resp, $end_stream_info);
$resp = $stream_init_info;