phpreflection

Get class name from file


I have a php file which contains only one class. how can I know what class is there by knowing the filename? I know I can do something with regexp matching but is there a standard php way? (the file is already included in the page that is trying to figure out the class name).


Solution

  • There are multiple possible solutions to this problem, each with their advantages and disadvantages. Here they are, it's up to know to decide which one you want.

    Tokenizer

    This method uses the tokenizer and reads parts of the file until it finds a class definition.

    Advantages

    Disadvantages

    Code

    $fp = fopen($file, 'r');
    $class = $buffer = '';
    $i = 0;
    while (!$class) {
        if (feof($fp)) break;
    
        $buffer .= fread($fp, 512);
        $tokens = token_get_all($buffer);
    
        if (strpos($buffer, '{') === false) continue;
    
        for (;$i<count($tokens);$i++) {
            if ($tokens[$i][0] === T_CLASS) {
                for ($j=$i+1;$j<count($tokens);$j++) {
                    if ($tokens[$j] === '{') {
                        $class = $tokens[$i+2][1];
                    }
                }
            }
        }
    }
    

    Regular expressions

    Use regular expressions to parse the beginning of the file, until a class definition is found.

    Advantages

    Disadvantages

    Code

    $fp = fopen($file, 'r');
    $class = $buffer = '';
    $i = 0;
    while (!$class) {
        if (feof($fp)) break;
    
        $buffer .= fread($fp, 512);
        if (preg_match('/class\s+(\w+)(.*)?\{/', $buffer, $matches)) {
            $class = $matches[1];
            break;
        }
    }
    

    Note: The regex can probably be improved, but no regex alone can do this perfectly.

    Get list of declared classes

    This method uses get_declared_classes() and look for the first class defined after an include.

    Advantages

    Disadvantages

    Code

    $classes = get_declared_classes();
    include 'test2.php';
    $diff = array_diff(get_declared_classes(), $classes);
    $class = reset($diff);
    

    Note: You cannot simply do end() as others suggested. If the class includes another class, you will get a wrong result.


    This is the Tokenizer solution, modified to include a $namespace variable containing the class namespace, if applicable:

    $fp = fopen($file, 'r');
    $class = $namespace = $buffer = '';
    $i = 0;
    while (!$class) {
        if (feof($fp)) break;
    
        $buffer .= fread($fp, 512);
        $tokens = token_get_all($buffer);
    
        if (strpos($buffer, '{') === false) continue;
    
        for (;$i<count($tokens);$i++) {
            if ($tokens[$i][0] === T_NAMESPACE) {
                for ($j=$i+1;$j<count($tokens); $j++) {
                    if ($tokens[$j][0] === T_STRING) {
                         $namespace .= '\\'.$tokens[$j][1];
                    } else if ($tokens[$j] === '{' || $tokens[$j] === ';') {
                         break;
                    }
                }
            }
    
            if ($tokens[$i][0] === T_CLASS) {
                for ($j=$i+1;$j<count($tokens);$j++) {
                    if ($tokens[$j] === '{') {
                        $class = $tokens[$i+2][1];
                    }
                }
            }
        }
    }
    

    Say you have this class:

    namespace foo\bar {
        class hello { }
    }
    

    ...or the alternative syntax:

    namespace foo\bar;
    class hello { }
    

    You should have the following result:

    var_dump($namespace); // \foo\bar
    var_dump($class);     // hello
    

    You could also use the above to detect the namespace a file declares, regardless of it containing a class or not.