Skip to content

Applied-Machine-Learning-2021/final-project-the-boston-project

Repository files navigation

final-project

National Action Council for Minorities in Engineering(NACME) Google Applied Machine Learning Intensive (AMLI) at the University of Kentucky

Developed by:

Machine Learning Advisor:

Other Advisors:

Background

Our project was originally started by Dr.Sesma, Doctor of Anthropology and Assistant Professor at the University of Kentucky, with specializations in African Diaspora and historical archaeology. For those who are not aware of what archaeology is, it is the study of human history and prehistory through the excavation of sites and the analysis of artifacts which include but are not limited to maps and diaries, photographs or pictures, and oral histories, along with physical remains from the deceased. One of the most common difficulties about archeology is that it is a destructive science. Once an area is excavated (where earth is moved carefully and systematically from (an area) in order to find buried remains), we can’t put things back the way we found them. Many of the sites have been connected to marginalized communities, especially from African-American communities, which have historically been destroyed in the process of countless development, expansion projects, or even for quests of truth. This eventually leads to question: "How can we best mitigate the damage archeologists do when excavating unknown sites and digging into the past?"

In efforts to solve this problem our team took up the task of building a Convolutional Neural Network (CNN) that determines whether or not an area of land containes a cemetery so archeologists can plan properly and mitigate damage through escavation. We used TensorFlow and natural color images from the kyfromabove website to train our model with a sigmoid activation on the last layer of the model, and a 3 x 3 kenel_size.

Usage instructions

Upload CNN_.ipynb file into Google colab

For this line to work properly you need to have aerial images uploded into your Google drive

drive.mount('/content/drive')

Uses Keras DirectoryIterator

This function uses the keras open library to form a variable to train our CNN model. This chooses the specified directory that will eventually train the model to make predictions.

train_image_iterator = tf.keras.preprocessing.image.DirectoryIterator(
    target_size=(100, 100),
    directory=train_dir,
    batch_size=128,
    image_data_generator=None)

CNN features

These are the features that used to construct the CNN. We chose to use a 3x3 kernal size and sigmoid activation on the layer because we are dealing with binary classification.

model = tf.keras.Sequential([
    tf.keras.layers.Conv2D(16, 3, padding='same', activation='relu',
                           input_shape=(100, 100, 3)),
    tf.keras.layers.MaxPooling2D(),
    tf.keras.layers.Conv2D(32, 3, padding='same', activation='relu'),
    tf.keras.layers.MaxPooling2D(),
    tf.keras.layers.Conv2D(64, 3, padding='same', activation='relu'),
    tf.keras.layers.MaxPooling2D(),
    tf.keras.layers.Flatten(),
    tf.keras.layers.Dense(512, activation='relu'),
    tf.keras.layers.Dense(2, activation='sigmoid')
])

Model features

Compiling and training the model. 100 epochs are used to train based on the file size.

model.compile(optimizer='adam',
              loss='binary_crossentropy',
              metrics=['accuracy'])


history = model.fit(
    train_image_iterator,
    epochs=100,
    shuffle=True
)

Exploratory Data Analysis \ Main_(LiDAR_and_point_cloud_testing)

Before we decided to use the images to train our model, we planned on using the lidar data we gathered from the kyfromabove website. First we downloaded one of the zipfiles from the the site and uncompressed the .laz file from it into a .txt file using laszip.exe. We opend the file and found that it containg seven to eight million x, y and z coordinates. we decided that we could train our CNN model using this data by creating an image that would use the x and y values as the points place in a matrix and the z value was the gray color chanel values that spot holds. As we were looking at the list of coordinates we took note that they were not ordered by x or y and were randomly placed. To fix this issue so that we could feed the data to our model we tried to create a structured array of x’s and y’s that were evenly spaced between the x and y minimum and maximum. Then we would take all of the data points and place them into their corresponding bins and find the average z value of all the points that belonged to each bin.

First we uploaded one of the .txt files into Google Colab and created a function that would open the file and placed the x y and z coordinates it contained into a pandas dataframe.

def open_lidar(filename, verbose=False):
    """Method for opening LiDAR text files and handling possible line errors
    """
    # open file and read in each line
    with open(filename, "r") as f:
        lines = f.readlines()

    # iterate through lines to cast to lists of floats
    new_lines = list()
    for line in lines:

        # in case file is corrupt 
        try:
            new_lines.append([float(val) for val in line.split()])

        except Exception as e:
            if verbose:  # printing only if verbose, ignore otherwise
                print(line)

    # convert nested list to pandas dataframe
    new_lines = pd.DataFrame(new_lines)
    new_lines = new_lines.rename(columns={0: "x", 1: "y", 2: "z"})

    return new_lines

Then we calculated the maximum, minimum and scale for the x y and z values and placed these statistics into a dataframe. The scales were calculated by subtracting the minimum from the maximum of the x, y and z values.

#calculated the max values, the min, the scale(max-min) and the median value 
max = lidar_df.max() 
min = lidar_df.min()

# organized the data into a list to be converted into a dataframe  
lidar_stats = pd.DataFrame(
    [min, max, max - min],
    index=["min", "max", "scale"]
)

Next were created a dataframe with columns that were numbers evenly space between the minimum nad maximum x and indecies that were numbers evenly space between the minimum and maximum y.

side = 2048
increment = side / lidar_stats.loc["scale"]
point_grid = pd.DataFrame(
    index=[y for y in np.arange(lidar_stats["y"]["min"], lidar_stats["y"]["max"], increment["y"])],
    columns=[x for x in np.arange(lidar_stats["x"]["min"], lidar_stats["x"]["max"], increment["x"])],
)

lastly we created a function that would take in an x, y and list of nearby coordinated and calculated the average z values with the x and y of each 3d coordinated in the list acting as a weight. Then we used a for-loop to travese the dataframe and gather all of the lidar points that would belong in that cell.

#This function will help form a LiDAR grid with a proper set of consistent points
def calc_elevation(x, y, nearby_data_points):
  """Helper method for calculating the elevation based off nearby points.
  """

  # if not nearby_data_points:
  #   return None

  distances = [((val[0] - x)**2 + (val[1] - y)**2)**0.5 for val in nearby_data_points]
  total_distance = len(distances)
  return sum([val[2] * (distances[index]/total_distance) for index, val in enumerate(nearby_data_points)])


gridded_df = pd.DataFrame(columns=np.arange(lidar_stats["x"]["min"], lidar_stats["x"]["max"], increment["x"]))

for x in np.arange(lidar_stats["x"]["min"], lidar_stats["x"]["max"], increment["x"]):
  for y in np.arange(lidar_stats["y"]["min"], lidar_stats["y"]["max"], increment["y"]):
    values = lidar_df[
                  
                      (lidar_df["x"] >= x) & \
                      (lidar_df["x"] <= (x + increment["x"])) & \
                      (lidar_df["y"] >= y) & \
                      (lidar_df["y"] <= (y + increment["y"]))
                      ]
    delete_indexes = values.index
    lidar_df.drop(labels=delete_indexes, axis=0, inplace=True)

    
    gridded_df[x][y] = calc_elevation(x, y, values.values)

    

This code was not very efficient and took too long to run thus we came up with another way of grouping and analyzing the data

Dealing with the Lidar Data

In this colab we tried to create a visualization of the x,y,z coordinates of the lidar data within txt file converted from an laz file that we got from the kyfromeabove website. Then we attempted to create a heatmap to show the concentration of lidar data to check of there is a consistent and structured distribution of lidar data to possibly create a sliding window method to check for cemetaries in specific areas.

(make sure to download and upload the text file before running code)

This function allowed us to view the lidar data


#view sample of the lidar data 
from random import sample

data = open_lidar("N092E301.txt")

sample_data = data.sample(n=5000, random_state=1)


from mpl_toolkits import mplot3d
%matplotlib inline
import numpy as np
import matplotlib.pyplot as plt

fig = plt.figure()
ax = plt.axes(projection='3d')

ax.scatter3D(sample_data["x"], sample_data["y"], sample_data["z"])
ax.view_init(60, 35)

This code allowed us to group to group the Lidar coordinates that were nearby eachother that there were many groups with no points in them showing that there were many blank areas in the data.

height = 2048
width = 2048
i , j = 0, 0
x = x_min
y = y_min
new_x_increment = (x_max - x_min) / height
new_y_increment = (y_max - y_min) / height
# grouping function to be used by groupby method
def GroupLidarPoints(df, ind, col_1, col_2):
  # for the x values
  x = df[col_1].loc[ind]
  x_group = int((x - x_min) / x_increment) + 1

  # for the y values
  y = df[col_2].loc[ind]
  y_group = int((y - y_min) / y_increment)
  return f'group {(y_group * width) + x_group}'
# grouping the lidar data
grouped_lidar_df = full_data.groupby(lambda X: GroupLidarPoints(full_data, X, 'X', 'Y'))
grouped_lidar_df.count()

This code allowed us to create a heatmap of the lidar points.


#may take up to 20min to run
for x in range(full_data.shape[0]):
  x_dummy = full_data['X'].iloc[x]
  y_dummy = full_data['Y'].iloc[x]
  
  x_index = ((x_dummy - int(x_min)) / round(x_increment)) - 1
  y_index = ((y_dummy - int(y_min)) / round(y_increment)) - 1
  empty_df.iloc[int(y_index), int(x_index)] = empty_df.iloc[int(y_index), int(x_index)] + 1

# Heatmap of the distribution of LiDAR points

import seaborn as sns
ax = sns.heatmap(empty_df, cmap='coolwarm')
plt.show()

The heat map helped us understand the inconsistency in the capture of lidar from the kyfromabove website. This lead to our decision to use the images from the kyfromabove website as the data we would feed to our CNN model, instead of the images we would make from the x, y and z coordinates of the lidar data.

Pictures

image image image image

About

final-project-the-boston-project created by GitHub Classroom

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •