| 
									
										
										
										
											2001-11-23 14:55:57 +00:00
										 |  |  | /************************************************************************* | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | *									 * | 
					
						
							|  |  |  | *	 YAP Prolog 							 * | 
					
						
							|  |  |  | *									 * | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |  *	Yap Prolog was developed at NCCUP - Universidade do Porto	 * | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | *									 * | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |  * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 * | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | *									 * | 
					
						
							|  |  |  | ************************************************************************** | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | * File:		system.yap						 * | 
					
						
							|  |  |  | * Last rev:								 * | 
					
						
							|  |  |  | * mods:									 * | 
					
						
							|  |  |  | * comments:	Operating System Access built-ins			 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @file   system.yap | 
					
						
							|  |  |  |  * @author VITOR SANTOS COSTA <vsc@VITORs-MBP.lan> | 
					
						
							|  |  |  |  * @date   Wed Nov 18 01:23:45 2015 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-03 09:22:23 +00:00
										 |  |  | :- module(operating_system_support, | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  |     [ | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  |      datime/1, | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  |      delete_file/1, | 
					
						
							|  |  |  |      delete_file/2, | 
					
						
							|  |  |  | 	directory_files/2, | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  | 	directory_map/2, | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 	environ/2, | 
					
						
							|  |  |  | 	exec/3, | 
					
						
							| 
									
										
										
										
											2016-02-03 09:22:23 +00:00
										 |  |  |      file_exists/2, | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 	file_property/2, | 
					
						
							|  |  |  | 	host_id/1, | 
					
						
							|  |  |  | 				     host_name/1, | 
					
						
							|  |  |  | 				     kill/1, | 
					
						
							|  |  |  | 				     md5/3, | 
					
						
							|  |  |  | 	pid/1, | 
					
						
							|  |  |  | 				     mktemp/2, | 
					
						
							|  |  |  | 	make_directory/1, | 
					
						
							|  |  |  | 	popen/3, | 
					
						
							| 
									
										
										
										
											2016-08-23 15:08:36 -05:00
										 |  |  |      read_link/3, | 
					
						
							| 
									
										
										
										
											2016-02-03 09:22:23 +00:00
										 |  |  |      rename_file/2, | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 	shell/0, | 
					
						
							|  |  |  | 	shell/1, | 
					
						
							|  |  |  | 	shell/2, | 
					
						
							|  |  |  | 	system/0, | 
					
						
							|  |  |  | 	system/1, | 
					
						
							|  |  |  | 	system/2, | 
					
						
							|  |  |  | 	mktime/2, | 
					
						
							|  |  |  | 	tmpnam/1, | 
					
						
							|  |  |  | 	tmp_file/2, | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  |     tmpdir/1, | 
					
						
							|  |  |  | 	wait/2, | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | 	working_directory/2 | 
					
						
							|  |  |  |           ]). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** @defgroup operating_system_support Operating System Functionality | 
					
						
							|  |  |  |  * @brief Portable Interaction with the OS, be it Unix, Linux, OSX, or Windows. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2015-01-04 23:58:23 +00:00
										 |  |  | @ingroup library | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | @{ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | YAP  provides a library of system utilities compatible with the | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | SICStus Prolog system library. This library extends and to some point | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | complements the functionality of Operating System access routines. The | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | library includes Unix/Linux and Win32 `C` code. They | 
					
						
							|  |  |  | are available through the `use_module(library(system))` command. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-18 15:06:25 +00:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | /** @pred file_property(+ _File_,? _Property_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The atom  _File_ corresponds to an existing file, and  _Property_ | 
					
						
							|  |  |  | will be unified with a property of this file. The properties are of the | 
					
						
							|  |  |  | form `type( _Type_)`, which gives whether the file is a regular | 
					
						
							|  |  |  | file, a directory, a fifo file, or of unknown type; | 
					
						
							|  |  |  | `size( _Size_)`, with gives the size for a file, and | 
					
						
							|  |  |  | `mod_time( _Time_)`, which gives the last time a file was | 
					
						
							|  |  |  | modified according to some Operating System dependent | 
					
						
							|  |  |  | timestamp; `mode( _mode_)`, gives the permission flags for the | 
					
						
							|  |  |  | file, and `linkto( _FileName_)`, gives the file pointed to by a | 
					
						
							|  |  |  | symbolic link. Properties can be obtained through backtracking: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  |    ?- file_property('Makefile',P). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | P = type(regular) ? ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | P = size(2375) ? ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | P = mod_time(990826911) ? ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | no | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | /** @pred host_id(- _Id_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Unify  _Id_ with an identifier of the current host. YAP uses the | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | `hostid` function when available, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | /** @pred host_name(- _Name_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Unify  _Name_ with a name for the current host. YAP uses the | 
					
						
							|  |  |  | `hostname` function in Unix systems when available, and the | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | `GetComputerName` function in WIN32 systems. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | /** @pred mktemp( _Spec_,- _File_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Direct interface to `mktemp`: given a  _Spec_, that is a file | 
					
						
							|  |  |  | name with six  _X_ to it, create a file name  _File_. Use | 
					
						
							|  |  |  | tmpnam/1 instead. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							|  |  |  | /** @pred mktime(+_Datime_, - _Seconds_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The `mktime/2` procedure receives a term of the form _datime(+ _Year_, | 
					
						
							|  |  |  | + _Month_, + _DayOfTheMonth_, + _Hour_, + _Minute_, + _Second_)_ and | 
					
						
							| 
									
										
										
										
											2016-02-03 09:22:23 +00:00
										 |  |  |   returns the number of _Seconds_ elapsed since 00:00:00 on January 1, | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 1970, Coordinated Universal Time (UTC).  The user provides information | 
					
						
							|  |  |  | on _Year_, _Month_, _DayOfTheMonth_, _Hour_, _Minute_, and | 
					
						
							|  |  |  | _Second_. The _Hour_ is given on local time. This function uses the | 
					
						
							|  |  |  | WIN32 `GetLocalTime` function or the Unix `mktime` function. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  |    ?- mktime(datime(2001,5,28,15,29,46),X). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | X = 991081786 ? ; | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | /** @pred pid(- _Id_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Unify  _Id_ with the process identifier for the current | 
					
						
							|  |  |  | process. An interface to the <tt>getpid</tt> function. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-23 15:08:36 -05:00
										 |  |  | /** @pred read_link(+ SymbolicLink, -Link, -NewPath) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Follow a _SymbolicLink_, and obtain the actual _Link_ and the target _newPath_). This predicate uses the | 
					
						
							|  |  |  | `C` built-in function `readlink` and is not yet operational in WIN32. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | /** @pred shell | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Start a new shell and leave YAP in background until the shell | 
					
						
							|  |  |  | completes. YAP uses the shell given by the environment variable | 
					
						
							|  |  |  | `SHELL`. In WIN32 environment YAP will use `COMSPEC` if | 
					
						
							|  |  |  | `SHELL` is undefined. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							|  |  |  | /** @pred shell(+ _Command_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Execute command  _Command_ under a new shell. YAP will be in | 
					
						
							|  |  |  | background until the command completes. In Unix environments YAP uses | 
					
						
							|  |  |  | the shell given by the environment variable `SHELL` with the option | 
					
						
							|  |  |  | `" -c "`. In WIN32 environment YAP will use `COMSPEC` if | 
					
						
							|  |  |  | `SHELL` is undefined, in this case with the option `" /c "`. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							|  |  |  | /** @pred shell(+ _Command_,- _Status_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Execute command  _Command_ under a new shell and unify  _Status_ | 
					
						
							|  |  |  | with the exit for the command. YAP will be in background until the | 
					
						
							|  |  |  | command completes. In Unix environments YAP uses the shell given by the | 
					
						
							|  |  |  | environment variable `SHELL` with the option `" -c "`. In | 
					
						
							|  |  |  | WIN32 environment YAP will use `COMSPEC` if `SHELL` is | 
					
						
							|  |  |  | undefined, in this case with the option `" /c "`. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							|  |  |  | /** @pred system | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Start a new default shell and leave YAP in background until the shell | 
					
						
							|  |  |  | completes. YAP uses `/bin/sh` in Unix systems and `COMSPEC` in | 
					
						
							|  |  |  | WIN32. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | /** @pred tmp_file(+_Base_, - _File_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | Create a name for a temporary file.  _Base_ is an user provided | 
					
						
							|  |  |  | identifier for the category of file. The  _TmpName_ is guaranteed to | 
					
						
							|  |  |  | be unique. If the system halts, it will automatically remove all created | 
					
						
							|  |  |  | temporary files. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | /** @pred tmpnam(- _File_) | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Interface with  _tmpnam_: obtain a new, unique file name  _File_. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-11 14:06:57 -05:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-07 17:54:29 +00:00
										 |  |  | :- use_module(library(lists), [append/3]). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | :- load_foreign_files([sys], [], init_sys). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-22 23:25:21 +00:00
										 |  |  | :- dynamic tmp_file_sequence_counter/1. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | % time builtins | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  | /** | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  @pred datime(datime(- _Year_, - _Month_, - _DayOfTheMonth_, - _Hour_, - _Minute_, - _Second_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The datime/1 procedure returns the current date and time, with | 
					
						
							|  |  |  | information on  _Year_,  _Month_,  _DayOfTheMonth_, | 
					
						
							|  |  |  |  _Hour_,  _Minute_, and  _Second_. The  _Hour_ is returned | 
					
						
							|  |  |  | on local time. This function uses the WIN32 | 
					
						
							|  |  |  | `GetLocalTime` function or the Unix `localtime` function. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  |    ?- datime(X). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | X = datime(2001,5,28,15,29,46) ? | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | datime(X) :- | 
					
						
							|  |  |  | 	datime(X, Error), | 
					
						
							| 
									
										
										
										
											2015-10-05 10:32:31 +01:00
										 |  |  | 	handle_system_internal(Error, off, datime(X)). | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-11 16:14:55 +00:00
										 |  |  | mktime(V, A) :- var(V), !, | 
					
						
							|  |  |  | 	throw(error(instantiation_error,mktime(V,A))). | 
					
						
							|  |  |  | mktime(In,Out) :- | 
					
						
							|  |  |  | 	check_mktime_inp(In, mktime(In,Out)), | 
					
						
							|  |  |  | 	In = datime(Y,Mo,D,H,Mi,S), | 
					
						
							|  |  |  | 	mktime(Y, Mo, D, H, Mi, S, Out, Error), | 
					
						
							| 
									
										
										
										
											2015-10-05 10:32:31 +01:00
										 |  |  | 	handle_system_internal(Error, off, mktime(In,Out)). | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-11 16:14:55 +00:00
										 |  |  | check_mktime_inp(V, Inp) :- var(V), !, | 
					
						
							|  |  |  | 	throw(error(instantiation_error,Inp)). | 
					
						
							|  |  |  | check_mktime_inp(datime(Y,Mo,D,H,Mi,S), Inp) :- !, | 
					
						
							|  |  |  | 	check_int(Y, Inp), | 
					
						
							|  |  |  | 	check_int(Mo, Inp), | 
					
						
							|  |  |  | 	check_int(D, Inp), | 
					
						
							|  |  |  | 	check_int(H, Inp), | 
					
						
							|  |  |  | 	check_int(Mi, Inp), | 
					
						
							|  |  |  | 	check_int(S, Inp). | 
					
						
							|  |  |  | check_mktime_inp(T, Inp) :- | 
					
						
							|  |  |  | 	throw(error(domain_error(mktime,T),Inp)). | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-08-11 16:14:55 +00:00
										 |  |  | check_int(I, _) :- integer(I), !. | 
					
						
							|  |  |  | check_int(I, Inp) :- var(I), | 
					
						
							|  |  |  | 	throw(error(instantiation_error,Inp)). | 
					
						
							|  |  |  | check_int(I, Inp) :- | 
					
						
							|  |  |  | 	throw(error(type_error(integer,I),Inp)). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | % file operations | 
					
						
							| 
									
										
										
										
											2013-02-12 16:21:26 -06:00
										 |  |  | % file operations | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | /** @pred delete_file(+ _File_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The delete_file/1 procedure removes file  _File_. If | 
					
						
							|  |  |  |  _File_ is a directory, remove the directory <em>and all its subdirectories</em>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  |    ?- delete_file(x). | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | See delete_file/2 for a more flexible version. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2013-02-12 16:21:26 -06:00
										 |  |  | delete_file(IFile) :- | 
					
						
							|  |  |  | 	true_file_name(IFile, File), | 
					
						
							|  |  |  | 	delete_file(File, off, on, off). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | /** @pred delete_file(+ _File_,+ _Opts_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The `delete_file/2` procedure removes file  _File_ according to | 
					
						
							|  |  |  | options  _Opts_. These options are `directory` if one should | 
					
						
							|  |  |  | remove directories, `recursive` if one should remove directories | 
					
						
							|  |  |  | recursively, and `ignore` if errors are not to be reported. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This example is equivalent to using the delete_file/1 predicate: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  |   ?- delete_file(x, [recursive]). | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2013-02-12 16:21:26 -06:00
										 |  |  | delete_file(IFile, Opts) :- | 
					
						
							|  |  |  | 	true_file_name(IFile, File), | 
					
						
							|  |  |  | 	process_delete_file_opts(Opts, Dir, Recurse, Ignore, delete_file(File,Opts)), | 
					
						
							|  |  |  | 	delete_file(File, Dir, Recurse, Ignore). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | process_delete_file_opts(V, _, _, _, T) :- var(V), !, | 
					
						
							|  |  |  | 	throw(error(instantiation_error,T)). | 
					
						
							|  |  |  | process_delete_file_opts([], off, off, off, _) :- !. | 
					
						
							|  |  |  | process_delete_file_opts([V|_], _, _, _, T) :- var(V), !, | 
					
						
							|  |  |  | 	throw(error(instantiation_error,T)). | 
					
						
							|  |  |  | process_delete_file_opts([directory|Opts], on, Recurse, Ignore, T) :- !, | 
					
						
							|  |  |  | 	process_delete_file_opts(Opts, _, Recurse, Ignore, T). | 
					
						
							|  |  |  | process_delete_file_opts([recursive|Opts], Dir, on, Ignore, T) :- !, | 
					
						
							|  |  |  | 	process_delete_file_opts(Opts, Dir, _, Ignore, T). | 
					
						
							|  |  |  | process_delete_file_opts([ignore|Opts], Dir, Recurse, on, T) :- !, | 
					
						
							|  |  |  | 	process_delete_file_opts(Opts, Dir, Recurse, _, T). | 
					
						
							|  |  |  | process_delete_file_opts(Opts, _, _, _, T) :- | 
					
						
							|  |  |  | 	throw(error(domain_error(delete_file_option,Opts),T)). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | delete_file(IFile, Dir, Recurse, Ignore) :- | 
					
						
							|  |  |  | 	true_file_name(IFile, File), | 
					
						
							|  |  |  | 	file_property(File, Type, _, _, _Permissions, _, Ignore), | 
					
						
							|  |  |  | 	delete_file(Type, File, Dir, Recurse, Ignore). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | delete_file(N, File, _Dir, _Recurse, Ignore) :- number(N), !, % error. | 
					
						
							| 
									
										
										
										
											2015-10-05 10:32:31 +01:00
										 |  |  | 	handle_system_internal(N, Ignore, delete_file(File)). | 
					
						
							| 
									
										
										
										
											2013-02-12 16:21:26 -06:00
										 |  |  | delete_file(directory, File, Dir, Recurse, Ignore) :- | 
					
						
							| 
									
										
										
										
											2013-02-14 20:42:23 -06:00
										 |  |  | 	delete_directory(Dir, File, Recurse, Ignore), !. | 
					
						
							| 
									
										
										
										
											2013-02-12 16:21:26 -06:00
										 |  |  | delete_file(_, File, _Dir, _Recurse, Ignore) :- | 
					
						
							|  |  |  | 	unlink_file(File, Ignore). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | unlink_file(IFile, Ignore) :- | 
					
						
							|  |  |  | 	true_file_name(IFile, File), | 
					
						
							|  |  |  | 	unlink(File, N), | 
					
						
							| 
									
										
										
										
											2015-10-05 10:32:31 +01:00
										 |  |  | 	handle_system_internal(N, Ignore, delete_file(File)). | 
					
						
							| 
									
										
										
										
											2013-02-12 16:21:26 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | delete_directory(on, File, _Recurse, Ignore) :- | 
					
						
							|  |  |  | 	rm_directory(File, Ignore). | 
					
						
							|  |  |  | delete_directory(off, File, Recurse, Ignore) :- | 
					
						
							|  |  |  | 	delete_directory(Recurse, File, Ignore). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | rm_directory(File, Ignore) :- | 
					
						
							|  |  |  | 	rmdir(File, Error), | 
					
						
							| 
									
										
										
										
											2015-10-05 10:32:31 +01:00
										 |  |  | 	handle_system_internal(Error, Ignore, delete_file(File)). | 
					
						
							| 
									
										
										
										
											2013-02-12 16:21:26 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | delete_directory(on, File, Ignore) :- | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | 	directory_files(File, FileList), | 
					
						
							| 
									
										
										
										
											2013-02-12 16:21:26 -06:00
										 |  |  | 	path_separator(D), | 
					
						
							|  |  |  | 	atom_concat(File, D, FileP), | 
					
						
							|  |  |  | 	delete_dirfiles(FileList, FileP, Ignore), | 
					
						
							|  |  |  | 	rmdir(File, Ignore). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | delete_dirfiles([], _, _). | 
					
						
							|  |  |  | delete_dirfiles(['.'|Fs], File, Ignore) :- !, | 
					
						
							|  |  |  | 	delete_dirfiles(Fs, File, Ignore). | 
					
						
							|  |  |  | delete_dirfiles(['..'|Fs], File, Ignore) :- !, | 
					
						
							|  |  |  | 	delete_dirfiles(Fs, File, Ignore). | 
					
						
							|  |  |  | delete_dirfiles([F|Fs], File, Ignore) :- | 
					
						
							|  |  |  | 	atom_concat(File,F,TrueF), | 
					
						
							|  |  |  | 	delete_file(TrueF, off, on, Ignore), | 
					
						
							|  |  |  | 	delete_dirfiles(Fs, File, Ignore). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | handle_system_internal(Error, _Ignore, _G) :- var(Error), !. | 
					
						
							|  |  |  | handle_system_internal(Error, off, G) :- atom(Error), !, | 
					
						
							|  |  |  | 	throw(error(system_internal(Error),G)). | 
					
						
							|  |  |  | handle_system_internal(Error, off, G) :- | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 	error_message(Error, Message), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	throw(error(system_internal(Message),G)). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | handle_system_internal(Error, _Id, _Ignore, _G) :- var(Error), !. | 
					
						
							| 
									
										
										
										
											2015-10-05 10:32:31 +01:00
										 |  |  | handle_system_internal(Error, _SIG, off, G) :- integer(Error), !, | 
					
						
							| 
									
										
										
										
											2015-04-24 10:03:44 -06:00
										 |  |  | 	error_message(Error, Message), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	throw(error(system_internal(Message),G)). | 
					
						
							|  |  |  | handle_system_internal(signal, SIG, off, G) :- !, | 
					
						
							|  |  |  |         throw(error(system_internal(child_signal(SIG)),G)). | 
					
						
							|  |  |  | handle_system_internal(stopped, SIG, off, G) :- | 
					
						
							|  |  |  |         throw(error(system_internal(child_stopped(SIG)),G)). | 
					
						
							| 
									
										
										
										
											2015-04-22 14:19:14 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-17 18:38:11 +00:00
										 |  |  | file_property(IFile, type(Type)) :- | 
					
						
							|  |  |  | 	true_file_name(IFile, File), | 
					
						
							| 
									
										
										
										
											2002-06-11 05:30:05 +00:00
										 |  |  | 	file_property(File, Type, _Size, _Date, _Permissions, _LinkName). | 
					
						
							| 
									
										
										
										
											2006-05-17 18:38:11 +00:00
										 |  |  | file_property(IFile, size(Size)) :- | 
					
						
							|  |  |  | 	true_file_name(IFile, File), | 
					
						
							| 
									
										
										
										
											2002-06-11 05:30:05 +00:00
										 |  |  | 	file_property(File, _Type, Size, _Date, _Permissions, _LinkName). | 
					
						
							| 
									
										
										
										
											2006-05-17 18:38:11 +00:00
										 |  |  | file_property(IFile, mod_time(Date)) :- | 
					
						
							|  |  |  | 	true_file_name(IFile, File), | 
					
						
							| 
									
										
										
										
											2002-06-11 05:30:05 +00:00
										 |  |  | 	file_property(File, _Type, _Size, Date, _Permissions, _LinkName). | 
					
						
							| 
									
										
										
										
											2006-05-17 18:38:11 +00:00
										 |  |  | file_property(IFile, mode(Permissions)) :- | 
					
						
							|  |  |  | 	true_file_name(IFile, File), | 
					
						
							| 
									
										
										
										
											2002-06-11 05:30:05 +00:00
										 |  |  | 	file_property(File, _Type, _Size, _Date, Permissions, _LinkName). | 
					
						
							| 
									
										
										
										
											2006-05-17 18:38:11 +00:00
										 |  |  | file_property(IFile, linkto(LinkName)) :- | 
					
						
							|  |  |  | 	true_file_name(IFile, File), | 
					
						
							| 
									
										
										
										
											2002-06-11 05:30:05 +00:00
										 |  |  | 	file_property(File, _Type, _Size, _Date, _Permissions, LinkName), | 
					
						
							|  |  |  | 	atom(LinkName). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | file_property(File, Type, Size, Date, Permissions, LinkName) :- | 
					
						
							|  |  |  | 	file_property(File, Type, Size, Date, Permissions, LinkName, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, file_property(File)). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-18 16:41:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | /** @pred environ(? _EnvVar_,+ _EnvValue_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Unify environment variable  _EnvVar_ with its value  _EnvValue_, | 
					
						
							|  |  |  | if there is one. This predicate is backtrackable in Unix systems, but | 
					
						
							|  |  |  | not currently in Win32 configurations. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  |    ?- environ('HOME',V). | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  | V = 'C:\\cygwin\\home\\administrator' ? | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | ~~~~~ | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  | _EnvVar_ may be bound to an atom, or just be | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |   unbound. In the latter case environ/2 will enumerate over all | 
					
						
							|  |  |  |   environment variables. | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | environ(Na,Val) :- var(Na), !, | 
					
						
							|  |  |  | 	environ_enum(0,I), | 
					
						
							|  |  |  | 	( p_environ(I,S) -> environ_split(S,SNa,SVal) ; !, fail ), | 
					
						
							|  |  |  | 	atom_codes(Na, SNa), | 
					
						
							|  |  |  | 	atom_codes(Val, SVal). | 
					
						
							|  |  |  | environ(Na,Val) :- atom(Na), !, | 
					
						
							|  |  |  | 	bound_environ(Na, Val). | 
					
						
							|  |  |  | environ(Na,Val) :- | 
					
						
							| 
									
										
										
										
											2002-06-17 06:27:00 +00:00
										 |  |  | 	throw(error(type_error(atom,Na),environ(Na,Val))). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | bound_environ(Na, Val) :- var(Val), !, | 
					
						
							|  |  |  | 	getenv(Na,Val). | 
					
						
							|  |  |  | bound_environ(Na, Val) :- atom(Val), !, | 
					
						
							|  |  |  | 	putenv(Na,Val). | 
					
						
							|  |  |  | bound_environ(Na, Val) :- | 
					
						
							| 
									
										
										
										
											2002-06-17 06:27:00 +00:00
										 |  |  | 	throw(error(type_error(atom,Val),environ(Na,Val))). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | environ_enum(X,X). | 
					
						
							|  |  |  | environ_enum(X,X1) :- | 
					
						
							|  |  |  | 	Xi is X+1, | 
					
						
							|  |  |  | 	environ_enum(Xi,X1). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | environ_split([61|SVal], [], SVal) :- !. | 
					
						
							|  |  |  | environ_split([C|S],[C|SNa],SVal) :- | 
					
						
							|  |  |  | 	environ_split(S,SNa,SVal). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-27 17:37:42 +01:00
										 |  |  | /** @pred exec(+ Command, StandardStreams, -PID) | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |  * Execute command _Command_ with its standard streams connected to the | 
					
						
							|  |  |  |  * list [_InputStream_, _OutputStream_, _ErrorStream_]. A numeric | 
					
						
							|  |  |  |  * identifier to the process that executes the command is returned as | 
					
						
							|  |  |  |  * _PID_. The command is executed by the default shell `bin/sh -c` in | 
					
						
							|  |  |  |  * Unix. | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |  * The following example demonstrates the use of exec/3 to send a | 
					
						
							|  |  |  |  * command and process its output: | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |  * ~~~~~ | 
					
						
							|  |  |  |   go :- | 
					
						
							|  |  |  |      exec(ls,[std,pipe(S),null],P), | 
					
						
							|  |  |  |      repeat, | 
					
						
							|  |  |  |      get0(S,C), | 
					
						
							|  |  |  |      (C = -1, close(S) ! ; put(C)). | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |  * The streams may be one of standard stream, `std`, null stream, | 
					
						
							|  |  |  |  * `null`, or `pipe(S)`, where  _S_ is a pipe stream. Note | 
					
						
							|  |  |  |  * that it is up to the user to close the pipe. | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | exec(Command, [StdIn, StdOut, StdErr], PID) :- | 
					
						
							|  |  |  | 	G = exec(Command, [StdIn, StdOut, StdErr], PID), | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 	check_command_with_default_shell(Command, TrueCommand, G), | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | 	process_inp_stream_for_exec(StdIn, In, G, [], L1), | 
					
						
							|  |  |  | 	process_out_stream_for_exec(StdOut, Out, G, L1, L2), | 
					
						
							|  |  |  | 	process_err_stream_for_exec(StdErr, Err, G, L2, L3), | 
					
						
							|  |  |  | 	( exec_command(TrueCommand, In, Out, Err, PID, Error) -> true ; true ), | 
					
						
							|  |  |  | 	close_temp_streams(L3), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, G). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | process_inp_stream_for_exec(Error, _, G, L, L) :- var(Error), !, | 
					
						
							| 
									
										
										
										
											2002-01-14 22:26:53 +00:00
										 |  |  | 	close_temp_streams(L), | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 	throw(error(instantiation_error,G)). | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | process_inp_stream_for_exec(null, null, _, L, L) :- !. | 
					
						
							|  |  |  | process_inp_stream_for_exec(std, 0, _, L, L) :- !. | 
					
						
							| 
									
										
										
										
											2002-01-10 23:54:06 +00:00
										 |  |  | process_inp_stream_for_exec(pipe(ForWriting), ForReading, _, L, [ForReading|L]) :- var(ForWriting), !, | 
					
						
							| 
									
										
										
										
											2018-04-27 13:01:08 +01:00
										 |  |  | 	open_pipe_stream(ForReading, ForWriting). | 
					
						
							| 
									
										
										
										
											2002-01-10 23:54:06 +00:00
										 |  |  | process_inp_stream_for_exec(pipe(Stream), _, _, L, L) :- !, | 
					
						
							| 
									
										
										
										
											2015-11-05 17:27:20 +00:00
										 |  |  | 	stream_property(Stream, input). | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | process_inp_stream_for_exec(Stream, Stream, _, L, L) :- | 
					
						
							| 
									
										
										
										
											2015-11-05 17:27:20 +00:00
										 |  |  | 	stream_property(Stream, put). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | process_out_stream_for_exec(Error, _, G, L, L) :- var(Error), !, | 
					
						
							| 
									
										
										
										
											2002-01-14 22:26:53 +00:00
										 |  |  | 	close_temp_streams(L), | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 	throw(error(instantiation_error,G)). | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | process_out_stream_for_exec(null, null, _, L, L) :- !. | 
					
						
							|  |  |  | process_out_stream_for_exec(std, 1, _, L, L) :- !. | 
					
						
							| 
									
										
										
										
											2002-01-10 23:54:06 +00:00
										 |  |  | process_out_stream_for_exec(pipe(ForReading), ForWriting, _, L, [ForWriting|L]) :- var(ForReading), !, | 
					
						
							| 
									
										
										
										
											2018-04-27 13:01:08 +01:00
										 |  |  | 	open_pipe_stream(ForReading, ForWriting). | 
					
						
							| 
									
										
										
										
											2002-01-10 23:54:06 +00:00
										 |  |  | process_out_stream_for_exec(pipe(Stream), _, _, L, L) :- !, | 
					
						
							| 
									
										
										
										
											2015-11-05 17:27:20 +00:00
										 |  |  | 	stream_property(Stream, output). | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | process_out_stream_for_exec(Stream, Stream, _, L, L) :- | 
					
						
							| 
									
										
										
										
											2015-11-05 17:27:20 +00:00
										 |  |  | 	stream_property(Stream, output). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | process_err_stream_for_exec(Error, _, G, L, L) :- var(Error), !, | 
					
						
							| 
									
										
										
										
											2002-01-14 22:26:53 +00:00
										 |  |  | 	close_temp_streams(L), | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 	throw(error(instantiation_error,G)). | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | process_err_stream_for_exec(null, null, _, L, L) :- !. | 
					
						
							|  |  |  | process_err_stream_for_exec(std, 2, _, L, L) :- !. | 
					
						
							| 
									
										
										
										
											2002-01-10 23:54:06 +00:00
										 |  |  | process_err_stream_for_exec(pipe(ForReading), ForWriting, _, L, [ForWriting|L]) :- var(ForReading), !, | 
					
						
							| 
									
										
										
										
											2018-04-27 13:01:08 +01:00
										 |  |  | 	open_pipe_stream(ForReading, ForWriting). | 
					
						
							| 
									
										
										
										
											2002-01-10 23:54:06 +00:00
										 |  |  | process_err_stream_for_exec(pipe(Stream), Stream, _, L, L) :- !, | 
					
						
							| 
									
										
										
										
											2015-11-05 17:27:20 +00:00
										 |  |  | 	stream_property(Stream, output). | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | process_err_stream_for_exec(Stream, Stream, _, L, L) :- | 
					
						
							| 
									
										
										
										
											2015-11-05 17:27:20 +00:00
										 |  |  | 	stream_property(Stream, output). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | close_temp_streams([]). | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | close_temp_streams([S|Ss]) :- | 
					
						
							| 
									
										
										
										
											2012-12-07 08:07:30 +00:00
										 |  |  | 	close(S), | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | 	close_temp_streams(Ss). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-27 17:37:42 +01:00
										 |  |  | /** @pred popen( +Command, +TYPE, -Stream) | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |  * Provides the functionaluty of the Unix <tt>popen</tt> function. It | 
					
						
							|  |  |  |  * opens a process by creating a pipe, forking and invoking _Command_ on | 
					
						
							|  |  |  |  * the child process. Since a pipe is by definition unidirectional the | 
					
						
							|  |  |  |  * _Type_ argument may be `read` or `write`, not both. The stream should | 
					
						
							|  |  |  |  * be closed using close/1, there is no need for a special `pclose` | 
					
						
							|  |  |  |  * command. | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |  * The following example demonstrates the use of popen/3 to process the | 
					
						
							|  |  |  |  * output of a command, note that popen/3 works as a simplified interface | 
					
						
							|  |  |  |  * to the exec/3 command: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  | ~~~~~{.prolog} | 
					
						
							|  |  |  | ?- popen(ls,read,X),repeat, get0(X,C), (C = -1, ! ; put(C)). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | X = 'C:\\cygwin\\home\\administrator' ? | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The implementation of popen/3 relies on exec/3. | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  |  * | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | popen(Command, read, Stream) :- | 
					
						
							|  |  |  | 	exec(Command, [std,pipe(Stream),std], Stream). | 
					
						
							|  |  |  | popen(Command, write, Stream) :- | 
					
						
							|  |  |  | 	exec(Command, [pipe(Stream),std,std], Stream). | 
					
						
							| 
									
										
										
										
											2011-03-08 00:07:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | check_command_with_default_shell(Com, ComF, G) :- | 
					
						
							|  |  |  | 	check_command(Com, G), | 
					
						
							|  |  |  | 	os_command_postprocess(Com, ComF). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | % make sure that Windows executes the command from $COMSPEC. | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | os_command_postprocess(Com, ComF) :- win, !, | 
					
						
							|  |  |  | 	atom_codes(Com, SC), | 
					
						
							|  |  |  | 	append(" /c ", SC, SC1), | 
					
						
							|  |  |  | 	getenv('COMSPEC', Shell0), | 
					
						
							|  |  |  | 	atom_codes(Shell0, Codes), | 
					
						
							|  |  |  | 	append(Codes, SC1, SCF), | 
					
						
							|  |  |  | 	atom_codes(ComF, SCF). | 
					
						
							|  |  |  | os_command_postprocess(Com, Com). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | check_command(Com, G) :- var(Com), !, | 
					
						
							|  |  |  | 	throw(error(instantiation_error,G)). | 
					
						
							| 
									
										
										
										
											2001-05-22 20:56:43 +00:00
										 |  |  | check_command(Com, _) :- atom(Com), !. | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | check_command(Com, G) :- | 
					
						
							| 
									
										
										
										
											2002-06-17 06:27:00 +00:00
										 |  |  | 	throw(error(type_error(atom,Com),G)). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-22 20:56:43 +00:00
										 |  |  | check_mode(Mode, _, G) :- var(Mode), !, | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 	throw(error(instantiation_error,G)). | 
					
						
							|  |  |  | check_mode(read, 0, _) :- !. | 
					
						
							|  |  |  | check_mode(write,1, _) :- !. | 
					
						
							|  |  |  | check_mode(Mode, G) :- | 
					
						
							| 
									
										
										
										
											2002-06-17 06:27:00 +00:00
										 |  |  | 	throw(error(domain_error(io_mode,Mode),G)). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-28 19:54:53 +00:00
										 |  |  | shell :- | 
					
						
							|  |  |  | 	G = shell, | 
					
						
							| 
									
										
										
										
											2001-06-08 14:52:54 +00:00
										 |  |  | 	get_shell0(FullCommand), | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | 	exec_command(FullCommand, 0, 1, 2, PID, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, G), | 
					
						
							| 
									
										
										
										
											2015-04-22 14:19:14 -06:00
										 |  |  | 	wait(PID, _Status, Error, Id), | 
					
						
							| 
									
										
										
										
											2016-04-18 16:41:30 +01:00
										 |  |  | 	handle_system_internal(Error, got(FullCommand, Id), off, G). | 
					
						
							| 
									
										
										
										
											2001-05-28 19:54:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | shell(Command) :- | 
					
						
							|  |  |  | 	G = shell(Command), | 
					
						
							|  |  |  | 	check_command(Command, G), | 
					
						
							| 
									
										
										
										
											2002-06-18 05:58:12 +00:00
										 |  |  | 	get_shell(Shell,Opt), | 
					
						
							| 
									
										
										
										
											2006-04-25 03:23:40 +00:00
										 |  |  | 	do_shell(Shell, Opt, Command, Status, Error), | 
					
						
							|  |  |  | 	Status = 0, | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, G). | 
					
						
							| 
									
										
										
										
											2001-05-28 19:54:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | shell(Command, Status) :- | 
					
						
							|  |  |  | 	G = shell(Command, Status), | 
					
						
							|  |  |  | 	check_command(Command, G), | 
					
						
							| 
									
										
										
										
											2002-06-18 05:58:12 +00:00
										 |  |  | 	get_shell(Shell,Opt), | 
					
						
							|  |  |  | 	do_shell(Shell, Opt, Command, Status, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, G). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-22 23:25:21 +00:00
										 |  |  | protect_command([], [0'"]). % " | 
					
						
							| 
									
										
										
										
											2002-06-17 05:39:53 +00:00
										 |  |  | protect_command([H|L], [H|NL]) :- | 
					
						
							|  |  |  | 	protect_command(L, NL). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-08 14:52:54 +00:00
										 |  |  | get_shell0(Shell) :- | 
					
						
							|  |  |  | 	getenv('SHELL', Shell), !. | 
					
						
							|  |  |  | get_shell0(Shell) :- | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | 	win, !, | 
					
						
							| 
									
										
										
										
											2002-01-10 21:42:29 +00:00
										 |  |  | 	getenv('COMSPEC', Shell). | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | get_shell0('/bin/sh'). | 
					
						
							| 
									
										
										
										
											2001-06-08 14:52:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-18 05:58:12 +00:00
										 |  |  | get_shell(Shell, '-c') :- | 
					
						
							|  |  |  | 	getenv('SHELL', Shell), !. | 
					
						
							|  |  |  | get_shell(Shell, '/c') :- | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | 	win, !, | 
					
						
							| 
									
										
										
										
											2002-06-18 05:58:12 +00:00
										 |  |  | 	getenv('COMSPEC', Shell). | 
					
						
							|  |  |  | get_shell('/bin/sh','-c'). | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  | /** | 
					
						
							|  |  |  |   * @pred  system(+ _S_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Passes command  _S_ to the Bourne shell (on UNIX environments) or the | 
					
						
							|  |  |  | current command interpreter in WIN32 environments. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |   * @pred  system | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Passes command  _S_ to the Bourne shell (on UNIX environments) or the | 
					
						
							|  |  |  | current command interpreter in WIN32 environments. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | system :- | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | 	default_shell(Command), | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | 	do_system(Command, _Status, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, system). | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | default_shell(Shell) :- win, !, | 
					
						
							|  |  |  | 	getenv('COMSPEC', Shell). | 
					
						
							|  |  |  | default_shell('/bin/sh'). | 
					
						
							| 
									
										
										
										
											2015-12-15 09:28:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  | /** @pred system(+ _Command_,- _Res_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Interface to `system`: execute command  _Command_ and unify | 
					
						
							|  |  |  |  _Res_ with the result. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | n*/ | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | system(Command, Status) :- | 
					
						
							|  |  |  | 	G = system(Command, Status), | 
					
						
							|  |  |  | 	check_command(Command, G), | 
					
						
							| 
									
										
										
										
											2001-06-29 19:30:25 +00:00
										 |  |  | 	do_system(Command, Status, Error), | 
					
						
							| 
									
										
										
										
											2006-04-25 03:23:40 +00:00
										 |  |  | 	Status = 0, | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, G). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-21 16:12:39 -06:00
										 |  |  | wait(PID,STATUS) :- var(PID), !, | 
					
						
							|  |  |  |  	throw(error(instantiation_error,wait(PID,STATUS))). | 
					
						
							|  |  |  | wait(PID,STATUS) :- integer(PID), !, | 
					
						
							| 
									
										
										
										
											2015-04-22 14:19:14 -06:00
										 |  |  |  	plwait(PID, STATUS, Error, _Detail), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  |  	handle_system_internal(Error, off, wait(PID,STATUS)). | 
					
						
							| 
									
										
										
										
											2015-04-21 16:12:39 -06:00
										 |  |  | wait(PID,STATUS) :- | 
					
						
							|  |  |  |  	throw(error(type_error(integer,PID),wait(PID,STATUS))). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | % host info | 
					
						
							|  |  |  | % | 
					
						
							|  |  |  | host_name(X) :- | 
					
						
							|  |  |  | 	host_name(X, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, host_name(X)). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | host_id(X) :- | 
					
						
							|  |  |  | 	host_id(X0, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, host_id(X)), | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 	number_codes(X0, S), | 
					
						
							|  |  |  | 	atom_codes(X, S). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | pid(X) :- | 
					
						
							|  |  |  | 	pid(X, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, pid(X)). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-21 16:12:39 -06:00
										 |  |  | kill(X,Y) :- | 
					
						
							|  |  |  | 	integer(X), integer(Y), !, | 
					
						
							|  |  |  | 	kill(X, Y, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, kill(X,Y)). | 
					
						
							| 
									
										
										
										
											2015-04-21 16:12:39 -06:00
										 |  |  | kill(X,Y) :- (var(X) ; var(Y)), !, | 
					
						
							|  |  |  | 	throw(error(instantiation_error,kill(X,Y))). | 
					
						
							|  |  |  | kill(X,Y) :- integer(X), !, | 
					
						
							|  |  |  | 	throw(error(type_error(integer,Y),kill(X,Y))). | 
					
						
							|  |  |  | kill(X,Y) :- | 
					
						
							|  |  |  | 	throw(error(type_error(integer,X),kill(X,Y))). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | mktemp(X,Y) :- var(X), !, | 
					
						
							|  |  |  | 	throw(error(instantiation_error,mktemp(X,Y))). | 
					
						
							|  |  |  | mktemp(X,Y) :- | 
					
						
							|  |  |  | 	atom(X), !, | 
					
						
							|  |  |  | 	mktemp(X, Y, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, mktemp(X,Y)). | 
					
						
							| 
									
										
										
										
											2001-05-21 20:08:10 +00:00
										 |  |  | mktemp(X,Y) :- | 
					
						
							|  |  |  | 	throw(error(type_error(atom,X),mktemp(X,Y))). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | tmpnam(X) :- | 
					
						
							|  |  |  | 	tmpnam(X, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  | 	handle_system_internal(Error, off, tmpnam(X)). | 
					
						
							| 
									
										
										
										
											2008-05-22 23:25:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-20 17:33:23 +01:00
										 |  |  | %%% Added from Theo, path_seperator is used to replace the c predicate dir_separator which is not OS aware | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | tmpdir(TmpDir):- | 
					
						
							|  |  |  |   tmpdir(Dir, Error), | 
					
						
							| 
									
										
										
										
											2015-09-29 23:51:09 +01:00
										 |  |  |   handle_system_internal(Error, off, tmpdir(Dir)), | 
					
						
							| 
									
										
										
										
											2010-12-20 17:33:23 +01:00
										 |  |  |   path_separator(D), | 
					
						
							|  |  |  |   (atom_concat(_, D, Dir) -> | 
					
						
							|  |  |  |     TmpDir = Dir | 
					
						
							|  |  |  |   ; | 
					
						
							|  |  |  |     atom_concat(Dir, D, TmpDir) | 
					
						
							|  |  |  |   ). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | path_separator('\\'):- | 
					
						
							|  |  |  |   win, !. | 
					
						
							|  |  |  | path_separator('/'). | 
					
						
							| 
									
										
										
										
											2016-08-23 15:08:36 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | read_link(P,D,F) :- | 
					
						
							|  |  |  | 	read_link(P, D), | 
					
						
							|  |  |  | 	absolute_file_name(D, [], F). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  | /** @pred rename_file(+ _OldFile_,+ _NewFile_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Create file  _OldFile_ to  _NewFile_. This predicate uses the | 
					
						
							|  |  |  | `C` built-in function `rename`. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | rename_file(F0, F) :- | 
					
						
							|  |  |  | 	rename_file(F0, F, Error), | 
					
						
							| 
									
										
										
										
											2018-04-27 13:01:08 +01:00
										 |  |  | 	handle_system_internal(Error, off, rename_file(F0, F)). | 
					
						
							| 
									
										
										
										
											2018-04-24 17:51:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | /** @pred directory_files(+ _Dir_,+ _List_) | 
					
						
							| 
									
										
										
										
											2018-06-07 18:05:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Given a directory  _Dir_,  directory_files/2 procedures a | 
					
						
							|  |  |  | listing of all files and directories in the directory: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  |     ?- directory_files('.',L), writeq(L). | 
					
						
							|  |  |  | ['Makefile.~1~','sys.so','Makefile','sys.o',x,..,'.'] | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | The predicates uses the `dirent` family of routines in Unix | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | environments, and `findfirst` in WIN32 through the system_library buil | 
					
						
							| 
									
										
										
										
											2018-06-07 18:05:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2018-06-14 11:27:43 +01:00
										 |  |  | directory_files(X,Y) :- | 
					
						
							| 
									
										
										
										
											2018-06-07 18:05:45 +01:00
										 |  |  |      list_directory(X,Y). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  | :- meta_predicate directory_map(+,1,-), | 
					
						
							|  |  |  | 	rb_apply(+,+,2,-). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** @pred directory_map(+ _Dir_, 1:_P_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Given a directory _Dir_, directory_map/2 visits all files in _Dir_, | 
					
						
							|  |  |  | and verifies whether `P(F)` holds, where _F_ is the file's absolute | 
					
						
							|  |  |  | path. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  |     ?- directory_map('.', process). | 
					
						
							|  |  |  | ~~~~~ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The predicates performs a left-recursive traversal. It does not protect against file system errors and it does not check for symbolic links. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2018-10-15 13:47:25 +01:00
										 |  |  | directory_map(D, P) :- | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  |         working_directory(_, D), | 
					
						
							|  |  |  | 	list_directory(D,L), | 
					
						
							| 
									
										
										
										
											2018-11-15 15:38:40 +00:00
										 |  |  | 	d_map(L,D, P). | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-15 15:38:40 +00:00
										 |  |  | d_map([],_,_). | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  | d_map(['.'|Fs],D, P) :- | 
					
						
							| 
									
										
										
										
											2018-10-15 13:47:25 +01:00
										 |  |  |     !, | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  |     d_map(Fs,D, P). | 
					
						
							|  |  |  | d_map(['..'|Fs],D, P) :- | 
					
						
							| 
									
										
										
										
											2018-10-15 13:47:25 +01:00
										 |  |  |     !, | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  |     d_map(Fs, D, P). | 
					
						
							| 
									
										
										
										
											2018-11-15 15:38:40 +00:00
										 |  |  | d_map([F|Fs], D, P) :- | 
					
						
							| 
									
										
										
										
											2018-10-12 14:57:05 +01:00
										 |  |  |     absolute_file_name( F, File, [prefix(D)] ), | 
					
						
							|  |  |  |     f_map(File, P), | 
					
						
							|  |  |  |     d_map(Fs, D, P). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | f_map(File, P) :- | 
					
						
							|  |  |  |      catch( file_property( File, type(directory) ), _, fail ), | 
					
						
							|  |  |  |      directory_map( File, P). | 
					
						
							|  |  |  | f_map(File, P) :- | 
					
						
							|  |  |  |      call(P,File). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-07 23:10:59 +01:00
										 |  |  | /** @} */ |