- 
                Notifications
    You must be signed in to change notification settings 
- Fork 8k
feat: add function num_cpus (formerly nproc) #11137
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: master
Are you sure you want to change the base?
Conversation
| Do you have an use case in mind ? | 
| 
 Thank you @devnexen 👍 | 
| I see. I m not against personally, I ll leave the decision to Mate on this one. | 
        
          
                ext/standard/basic_functions.c
              
                Outdated
          
        
      | } | ||
| #endif | ||
|  | ||
| RETURN_LONG(1); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is not really good way to inform about errors or being unsupported by the platform. It should probably error somehow if there is an error (either warning or exception) and similarly inform user that the function is not support or even not expose it at all if it is not supported.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is not really good way to inform about errors or being unsupported by the platform.
I understand. Do you think we can keep the default 0 now that windows is also supported? I assume the implementation should work for most OSs.
either warning or exception
What exception would fit here?
even not expose it at all if it is not supported.
Similar to sys_getloadavg.
We would expose the function for every OS with _SC_NPROCESSORS_ONLN and for windows.
The disadvantage is that a function_exists call is necessary before using the function.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Returning NULL might also be an option to allow cpu_cores() ?? 1 (or maybe implement it yourself for the given architecture). The signature will also make it more obvious that this can fail. A warning is hard to catch and handle (unless the return value is also distinguishable).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks 👍
NULL works for me and num_cpus() ?? 1 is very clean.
| Renamed the function to  | 
        
          
                ext/standard/basic_functions.c
              
                Outdated
          
        
      | /* }}} */ | ||
| #endif | ||
|  | ||
| PHP_FUNCTION(cpu_cores) | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nit: We might want to incorporate num, cpu_cores sounds like it might provide more information, like cpu_num_cores. But admittedly it doesn't sound as nice.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
True. I found https://docs.rs/num_cpus/latest/num_cpus/ and thought it was a good name.
| Renamed the function to  | 
| My personal opinion: I would love to see convenient OO APIs like what ext/random has instead of creating functions scattered here and there. The way I can imagine this is adding a  | 
| @kocsismate Unless there's some common configuration/parameter these functions would share, I don't see the point in hiding this function behind an object. A namespace is a different story, but I'm not ready for the bike-shedding 😜 | 
| @iluuu1994 I don't see it as "hiding", but grouping them together, so that the complete list of available functions for "server reflection" could be seen at once when someone starts to write  | 
| @kocsismate Grouping is what namespaces are for. 🙂  | 
| In terms of naming I agree that just single function is fine here (we don't really need OO API for this IMHO). The name  | 
| Hey 👋 Addressed the code review suggestions, had a brief look at Solaris/OpenBSD compatibility and rebased. | 
        
          
                ext/standard/basic_functions.c
              
                Outdated
          
        
      | { | ||
| ZEND_PARSE_PARAMETERS_NONE(); | ||
|  | ||
| #if defined(_SC_NPROCESSORS_ONLN) | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just NIT: the preprocesor macros are not usually indented in our code so if you could remove indent for them, it would be great.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure, they are much easier to spot when not indented.
| @kocsismate Are you still have an objection against the naming? Just checking if we need an RFC for this basically? @kesselb If there are no objections from @kocsismate, then it would be good to drop email to internals to see if there any objections in the next two weeks. If not, we could still get it to 8.3, otherwise the RFC would be required which would mean waiting for 8.4 due to discussion and voting period being both 2 weeks and feature freeze on July 18th. | 
| 
 To be honest, I'd be much happier if there was some structuring of the functionality which we add. I agree that the bike shedding can be extremely frustrating, but at least sometimes it improve things. Unfortunately, I'm still pretty much obsessed with my ReflectionServer idea, probably because I like the OO API of PHP much more than the procedural one: I always refer to ext/random as the best example for adding a new API. So the very least, it would make sense to think about the possible extensions of num_cpus? What kind of further info do we plan to provide about the environment in the future? If we want to stop here then ok, let's add a simple function no matter in which namespace. If we plan to add further related functionality then let's go for either a class (possibly with static methods) or with a separate namespace so that the choice is in line with our long term plans. As far as I noticed, PHP is evolving towards more and more object oriented APIs (ext/random, resource to object migrations, stop adding dual APIs with the preference of OO), so I think this topic is worth having a discussion at least internally. | 
| Hi, Sorry it took more than 2 years for me to follow up on this one 🙈 I agree it's worth revisiting this before adding a new function, to see if it fits with the current structure. The motivation for this PR was that there isn't a simple way to obtain the number of available CPU cores from PHP. For Linux, it's straightforward by reading /proc/cpuinfo, but on FreeBSD or Windows, it requires more work. There's a nice library for getting the CPU count (used by popular tools like PHP-CS-Fixer or Psalm) that supports different ways of obtaining the CPU count: https://packagist.org/packages/fidry/cpu-core-counter. In Nextcloud's Serverinfo app, we display various system information. Where possible, we rely on PHP's built-in functions (gethostname, net_get_interfaces, sys_getloadavg) because they work on both Linux and FreeBSD. @kocsismate asked about further environment details. In Serverinfo, we show: 
 These would benefit from being provided by ReflectionServer. Some are a bit niche though (like thermal zones). Could you please give me a brief update on whether this feature is still acceptable and what I need to do to move it forward? I'm also pinging @theofidry, the maintainer of cpu-core-counter, to see if he would use such a function. | 
| Re-reading the discussion, I think this should probably go through RFC as there was not a clear agreement about the API and there were various ideas. | 
| Thanks for the ping! For context the CPU counter library was developed as an effort to consolidate code that was copy/pasted left and right with slight modifications each time. From the perspective of that library, the desired information is: 
 From the PHP language perspective, I suspect there is a bit more information that could be provides. A naive first source of information I used was https://nodejs.org/api/os.html. I don't feel qualified to say what should really be in the language and what shouldn't, and certainly this should be a RFC. But for sure having an easier way to get info on the CPUs, other than trying to get the proc file or other depending of the architecture, would be nice! | 
| Providing an Os, System or Environment namespace to provide this kind of information would certainly make sense to me. I would like to see some plan for it, though. Things like https://www.php.net/manual/en/function.sys-getloadavg.php or https://www.php.net/manual/en/function.php-uname.php should also be available there (possibly with a cleaned up API). | 
| Could the name be shortened to  | 
Signed-off-by: Daniel Kesselberg <[email protected]>
| @theodorejb I don't mind the long name - it's nothing you're going to call at many different places in an application. | 
| /* }}} */ | ||
| #endif | ||
|  | ||
| PHP_FUNCTION(num_available_processors) | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'd just like to echo my feedback I just provided in PHP Internals here as well:
- 
I think "processor" is a bit too ambiguous. I'd use "cpu_core" instead. 
 Yes, technically that's not entirely accurate when hyper-threading is used, but in most cases it's not trivial to distinguish physical cores from logical cores anyway, and "cpu_cores" provides the most understandable abstraction for the vast majority of use cases: deciding how many parallel processes one should use for optimal use of the CPU.
- 
Is it necessary to add "available" as a disambiguator? 
 Are there future plans to add a function that provides the "total" or "unavailable" number of processors? If not, I'd just drop the "available" part.
- 
From a quick search in php-srcthere doesn't seem to be any existing function name starting withnum_. For the sake of consistency with the existing PHP functions, similar functionality in other languages, and following the principle of prefixing by primary concept, I suggest suffixing the function name with_countinstead.
tl;dr: I'd like to respectfully propose the following function name instead:
cpu_core_count()
This helps with the semantic grouping of the function, and (to me, at least) gives a more succinct and understandable description of the function.
| How about function num_cpus(bool $throw_on_error=true): ?boolThat way people can do  | 
| if (nprocs > 0) { | ||
| RETURN_LONG(nprocs); | ||
| } | ||
| #elif defined _WIN32 && ! defined __CYGWIN__ | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
| #elif defined _WIN32 && ! defined __CYGWIN__ | |
| #elif defined _WIN32 | 
The check for cygwin is superfluous because _SC_NPROCESSORS_ONLN  is defined: https://github.com/cygwin/cygwin/blob/2f43c4b625a1f72dc794144c799bfa4d9e812716/winsup/cygwin/sysconf.cc#L64
Returns the number of processors which are currently online.
The idea, to add a feature similar to nproc, comes from nextcloud/server#37921.
We use the number of processors to limit the parallel generation of thumbnails. Unfortunately, we forgot to implement a check if access to /proc/cpuinfo is allowed and /proc is not available for FreeBSD.
Our use case is probably not the best, but it is complicated to get the number of processors without shell_exec.
nprocis likely not a good name. The actual nproc implementation covers much more corner cases1.I was curious if I could add such a functionality with my limited c knowledge 🙈
If you like the idea and there is a chance, this patch could be accepted, I'm happy to spend more time on it.
Footnotes
https://github.com/coreutils/gnulib/blob/master/lib/nproc.c ↩