How can I detect if the user is on localhost in PHP?

前端 未结 9 1259
耶瑟儿~
耶瑟儿~ 2020-12-12 12:45

In other words, how can I tell if the person using my web application is on the server it resides on? If I remember correctly, PHPMyAdmin does something like this for securi

相关标签:
9条回答
  • 2020-12-12 13:16

    If you want to have a whitelist / allowlist that supports static IPs and dynamic names.

    For example:

    $whitelist = array("localhost", "127.0.0.1", "devel-pc.ds.com", "liveserver.com");
    if (!isIPWhitelisted($whitelist)) die();
    

    This way you could set a list of names/IPs that will be able (for sure) to be detected. Dynamic names add more flexibility for accessing from different points.

    You have two common options here, you could set a name in your local hosts file or you could just use one dynamic name provider that could be found anywhere.

    This function CACHES results because gethostbyname is a very slow function.

    For this pupose I've implemented this function:

    function isIPWhitelisted($whitelist = false)
    {
        if ( isset($_SESSION) && isset($_SESSION['isipallowed']) )
            { return $_SESSION['isipallowed'];  }
    
        // This is the whitelist
        $ipchecklist = array("localhost", "127.0.0.1", "::1");
        if ($whitelist) $ipchecklist = $whitelist;
    
        $iplist = false;
        $isipallowed = false;
    
        $filename = "resolved-ip-list.txt";
        $filename = substr(md5($filename), 0, 8)."_".$filename; // Just a spoon of security or just remove this line
    
        if (file_exists($filename))
        {
            // If cache file has less than 1 day old use it
            if (time() - filemtime($filename) <= 60*60*24*1)
                $iplist = explode(";", file_get_contents($filename)); // Read cached resolved ips
        }
    
        // If file was not loaded or found -> generate ip list
        if (!$iplist)
        {
            $iplist = array(); $c=0;
            foreach ( $ipchecklist as $k => $iptoresolve )
            {
                // gethostbyname: It's a VERY SLOW function. We really need to cache the resolved ip list
                $ip = gethostbyname($iptoresolve);
                if ($ip != "") $iplist[$c] = $ip;
                $c++;
            }
    
            file_put_contents($filename, implode(";", $iplist));
        }
    
        if (in_array($_SERVER['REMOTE_ADDR'], $iplist)) // Check if the client ip is allowed
            $isipallowed = true;
    
        if (isset($_SESSION)) $_SESSION['isipallowed'] = $isipallowed;
    
        return $isipallowed;
    }
    

    For better reliability you could replace the $_SERVER['REMOTE_ADDR'] for the get_ip_address() that @Pekka mentioned in his post as "this bounty question"

    0 讨论(0)
  • 2020-12-12 13:21

    Newer OS users (Win 7, 8) may also find it necessary to include an IPV6-format remote address in their whitelist array:

    $whitelist = array('127.0.0.1', "::1");
    
    if(!in_array($_SERVER['REMOTE_ADDR'], $whitelist)){
        // not valid
    }
    
    0 讨论(0)
  • 2020-12-12 13:30

    You can also use $_SERVER['REMOTE_ADDR'] for which IP address of the client requesting is given by the web server.

    $whitelist = array(
        '127.0.0.1',
        '::1'
    );
    
    if(!in_array($_SERVER['REMOTE_ADDR'], $whitelist)){
        // not valid
    }
    
    0 讨论(0)
  • 2020-12-12 13:31

    It doesn't seem you should use $_SERVER['HTTP_HOST'], because this is the value in http header, easily faked.

    You may use $_SERVER["REMOTE_ADDR"] too, this is the more secure value, but it is also possible to fake. This remote_addr is the address where Apache returns result to.

    0 讨论(0)
  • 2020-12-12 13:33

    As a complement, as a function...

    function isLocalhost($whitelist = ['127.0.0.1', '::1']) {
        return in_array($_SERVER['REMOTE_ADDR'], $whitelist);
    }
    
    0 讨论(0)
  • 2020-12-12 13:37

    I'm sorry but all these answers seem terrible to me. I would suggest rephrasing the question because in a sense all machines are "localhost".

    The question should be; How do I run different code paths depending on which machine it is executed on.

    In my opinion, the easiest way is to create a file called DEVMACHINE or whatever you want really and then simply check

    file_exists('DEVMACHINE')

    Remember to exclude this file when uploading to the live hosting environment!

    This solution is not depending on network configuration, it can not be spoofed and makes it easy to switch between running "live-code" and "dev-code".

    0 讨论(0)
提交回复
热议问题