This assignment will help you understand security mechanisms. You will be guided through the steps of creating a reference monitor using the security layer functionality in Repy V2. A reference monitor is an access control concept that refers to an abstract machine that mediates all access to objects by subjects. This can be used to allow, deny, or change the behavior of any set of calls. While not a perfect way of validating your reference monitor, it is useful to create test cases to see whether your security layer will work as expected. (The test cases may be turned in as part of the next assignment.)
This assignment is intended to reinforce concepts about access control and reference monitors in a hands-on manner.
In this assignment you will create a reference monitor that will, when asked to do so, protect files starting with private_ (and only those files). If the contents of a file begin with the characters “INV”, then a listfiles() call must not show that the file exists. If the contents of a file begin with the characters “PER” then a removefile() call must raise a RepyArgumentError. In cases not specified above, all calls must work identically to the RepyV2 API.
Three design paradigms are at work in this assignment: accuracy, efficiency, and security.
- Accuracy: The reference monitor you create should only stop the specified actions from being executed. All other actions should be allowed. For example, if a user tries to open or remove an invisible file, this should be allowed. If a user tries to delete a file “foo” that starts with the characters “PER” this is also allowed (since the file name does not start with private_).
- Efficiency: The reference monitor should use a minimum number of resources, so it does not slow down the system. This means you must not re-read the header of each private_ file every time there is a listfiles() or removefile() call. You will need to track the state of the files as the system operates, including the data that is written to files.
- Security: The attacker should not be able to circumvent the reference monitor and cause files that should not be displayed to be displayed or files that should not be removed to be removed.
Please note you must have Python 2.5, 2.6, or 2.7 to complete this assignment. Instructions on how to get Python for Windows can be found [InstallPythonOnWindows here]. If you are using Linux or a Mac it is likely you already have Python. In order to verify this, simply open a terminal and type python
. Please check its version on the initial prompt.
**Note:**If you are using Windows, you will need python in your path variables. A tutorial on adding variables to your path in Windows can be found at [http://www.computerhope.com/issues/ch000549.htm]
The preferred way to get Repy is ''installing from source''. For this, you check out the required Git repositories, and run a build script. You can always update the repos later, and rebuild, so that you get the latest stable version of the Repy runtime.
Here's how to do that. Assuming you are running on a Unixoid OS,
# Create a directory for the required Git repositories
mkdir SeattleTestbed
cd SeattleTestbed
# Check out the repos required for building Repy
git clone https://github.com/SeattleTestbed/repy_v2.git
# Prepare a build directory, and build into it
cd repy_v2/scripts
mkdir ~/path/to/build/dir
python initialize.py
python build.py ~/path/to/build/dir
Once the build script finished, ~/path/to/build/dir
contains a ready-to-use copy of the RepyV2 runtime!
(If you cannot install from source, [attachment:repyv2_commit_3499642.zip:wiki:EducationalAssignments/SecurityLayerPartOne zipfile] is a tarball including a pre-built runtime.)
Use the command found below in order to run Repy files:
python repy.py restrictions.default encasementlib.r2py [security_layer].r2py [program].r2py
Please note Repy files end in extension .r2py
.
In order to test whether or not these steps worked, please copy and paste the code found below for the sample security layer and sample attack.
- Sample security layer: [wiki:EducationalAssignments/ProtectFilePartOne#Abasicandinadequatedefense Abasicandinadequatedefense]
- Sample attack layer: [wiki:EducationalAssignments/ProtectFilePartOne#Testingyoursecuritylayer Testingyoursecuritylayer]
If you got an error, please go through the troubleshooting section found below.
If you can't get Repy files to run, some of the following common errors may have occurred:
- using
print
instead oflog
:
Repy is a subset of Python, but its syntax is slightly different. For example, Python's print
statement cannot be used; Repy has log
for that. For a full list of acceptable syntax please see [wiki:RepyV2API].
- command line errors:
files are missing: In the above command line call, you must have repy.py
, restrictions.default
, encasementlib.r2py
, the security layer and the program you want to run in the current working directory. If any or all of the above files are not in that directory then you will not be able to run Repy files.
Now that you have Repy and Python, you may need a refresher on how to use them. The following tutorials are excellent sources of information.
- Python tutorial: [http://docs.python.org/tutorial/]
- Seattle tutorial: [https://seattle.poly.edu/wiki/PythonVsRepy]
- List of RepyV2 API calls: [wiki:RepyV2API]
The following program is a basic and incomplete sample code for you to get an idea about writing security layer. Remember, you have no idea how the attacker will try to penetrate your security layer, so it is important that you leave nothing to chance!
Time to start coding! Let's inspect a basic security layer.
INVBUFFER = []
PERBUFFER = []
class SecureFile():
def __init__(self, file, filename):
self.file = file
self.filename = filename
def readat(self, sizelimit, offset):
return self.file.readat(sizelimit, offset)
def writeat(self, data, offset):
if offset == 0 and self.filename.startswith("private_"):
if data == "INV":
INVBUFFER.append(self.filename)
# remove the file from the PERBUFFER if it exists
try:
PERBUFFER.remove(self.filename)
except ValueError:
pass
elif data == "PER":
PERBUFFER.append(self.filename)
# remove the file from the INVBUFFER if it exists
try:
INVBUFFER.remove(self.filename)
except ValueError:
pass
return self.file.writeat(data, offset)
def close(self):
return self.file.close()
def secure_openfile(filename, create):
file = openfile(filename, create)
return SecureFile(file, filename)
def secure_removefile(filename):
if filename in PERBUFFER:
raise RepyArgumentError("Cannot delete this file!
n")
removefile(filename)
def secure_listfiles():
filelist = listfiles()
returnlist = []
for filename in filelist:
if filename in INVBUFFER:
continue
returnlist.append(filename)
return returnlist
# The code here sets up type checking and variable hiding for you. You should not need to change anything below here.
sec_file_def = {
"obj-type":SecureFile,
"name":"SecureFile",
"readat":{"type":"func","args":((int,long,type(None)),(int,long)),"exceptions":Exception,"return":str,"target":SecureFile.readat},
"writeat":{"type":"func","args":(str,(int,long)),"exceptions":Exception,"return":(None),"target":SecureFile.writeat},
"close":{"type":"func","args":None,"exceptions":None,"return":(bool,type(None)),"target":SecureFile.close}
}
CHILD_CONTEXT_DEF["openfile"] = {"type":"objc",
"args":(str,bool),
# any? why Exception in particular?
"exceptions":Exception,
"return":sec_file_def,
"target":secure_openfile
}
CHILD_CONTEXT_DEF["listfiles"] = {"type":"func",
"args":None,
"return":list,
"exceptions":"any",
"target":secure_listfiles
}
CHILD_CONTEXT_DEF["removefile"] = {"type":"func",
"args":(str,),
"return":None,
"exceptions":"any",
"target":secure_removefile
}
# Execute the user code
secure_dispatch_module()
In this part of the assignment you will pretend to be an attacker. Remember the attacker's objective is to bypass the restrictions or cause the security layer to act in a disallowed manner. By understanding how the attacker thinks, you will be able to write better security layers.
An example of an attack is found below:
if "private_testfile.txt" in listfiles():
removefile("private_testfile.txt")
# Open a file
myfile = openfile("private_testfile.txt", True)
# Put some data in the file
myfile.writeat("INV", 0)
if "private_testfile.txt" in listfiles():
log("ERROR! File not invisible.
n")
# Write into the file again
myfile.writeat("PER", 0)
#Close the file
myfile.close()
try:
removefile("private_testfile.txt")
except RepyArgumentError:
pass # There should be an exception here!
else:
log("ERROR! Allowed me to delete the file.
n")
Note: All attacks should be written as Repy V2 files, using the .r2py extension.
It is important to keep in mind that only lowercase file names are allowed. So in the above code, specifically:
# Open a file
myfile=openfile("look.txt",True)
look.txt is a valid file name, however Look.txt is not. Examples of other invalid files names are, [email protected], look/.txt, and look().txt. Essentially all non-alphanumeric characters are not allowed.
Finally, type the following commands at the terminal to run your security layer with your attack program
python repy.py restrictions.default encasementlib.r2py [security_layer].r2py [attack_program].r2py
Make sure you went through the "How to get RepyV2" section!
-
For a complete list of syntax in Repyv2 please visit: [wiki:RepyV2API]
-
The following link is an excellent source for information about security layers: [https://ssl.engineering.nyu.edu/papers/cappos_seattle_ccs_10.pdf]
-
It is possible to add multiple security layers to Repy, this may be useful for testing different mitigations separately. This is done with the following command at the terminal:
python repy.py restrictions.default encasementlib.r2py [security_layer1].r2py [security_layer2].r2py [security_layer3].r2py [program].r2py
-
**Your security layer should produce no output!! **
-
In repy log replaces print from python. This may be helpful when testing if Repy installed correctly.
For extra credit, make a second security layer which will prevent writing to a private_ file when the first and last three characters in the file match and are either "INV" or "PER". If the first three characters in a private_ file are "INV" and the last three are "PER", you should be able to write to the file.
Note: Do not submit this code inside your assignment file. Submit a separate copy for extra credit.
-
Turn in a repy file called reference_monitor_[netid].r2py with all letters in lowercase.
-
Never raise unexpected errors or produce any output. Your program must produce no output when run normally.
-
For extra credit turn in a second repy file called extra_credit_[netid].r2py You must turn in separate files for the normal assignment and extra credit