COMP20008-Data-Linkage

Comparison between Naive and Blocking for Data Linkage

COMP20008 Project 2 Part 1: Data Linkage

Marks: 11/12

Abt and Buy both have product databases. In order to perform joint market research they need to link the same products in their databases. Therefore the research team has manually linked a subset of the data in order to build a linkage algorithm to automate the remaining items to be linked.

In this project, two implementations of Data Linkage were carried out:

  1. Naive method
  2. Blocking method

All files and source codes are located in the /src folder.

Project Specification

abt.csv

abt_small.csv

buy.csv

buy_small.csv

Naive Method

Script

Libraries Used

import pandas as pd
import textdistance
from sklearn.preprocessing import MinMaxScaler
from sklearn.preprocessing import StandardScaler
import csv
import nltk
import re

Load CSV’s

buy_s=pd.read_csv('buy_small.csv',encoding = 'ISO-8859-1')
abt_s=pd.read_csv('abt_small.csv',encoding = 'ISO-8859-1')

buy_s=buy_s.loc[:,['idBuy','name','description','price']]
abt_s=abt_s.loc[:,['idABT','name','description','price']]

To process the two datasets, the csv’s were first imported into Pandas dataframes. The common and relevant columns are then extracted from both datasets, namely the ID, name, description and price of the products. Before data linkage can be done, pre-processing was performed on the price of the products to remove dollar signs using regular expressions. Then, each item from ‘abt_small.csv’ was used to loop with each item from ‘buy_small.csv’, to compare two items.

buy_s[['price']] = buy_s[['price']].replace('[\$,]','',regex=True).astype(float)
abt_s[['price']] = abt_s[['price']].replace('[\$,]','',regex=True).astype(float)

Common punctuations were first removed from their names, and they were converted into lower case and tokenized using word_tokenize from nltk. In choosing the similarity functions between names, token-based algorithms are more suitable for string of words than edit-based. Hence, cosine_normalized_similarity was used on two strings (from the textdistance library). The similarity between the names is then computed. We also extract the brand and model from the tokenized product name, as product name is observed to appear mostly in the first position and model name in the last (if any).

We then compare the brand and model and assign similarity scores respectively. The absolute price difference is also calculated and assigned scores. The scoring function f is

0.5*(name score) + 0.2*(model score) + 0.2*(brand score) + 0.1*(price score), ranges between [0,1].

Model scores and brand scores have default values of 0.2 and 0.0 and are assigned 1 respectively if brand matches or model (assumed to be alphanumerical characters) in one product can be found in another. Price score is 0.5 if $10 < price difference < $50, 1.0 if below range and 0.0 if above range. Similarity scores between one product from ‘buy’ and all products from ‘abt’ were ranked and the pair with the highest score was added into the match list if it exceeds the threshold θ1.

A list was also built to filter out products of ‘abt’ that have previously in a highly similar match (similarity > θ2) to reduce duplication. If the top ‘abt’ product in the loop has previously been in a highly similar match, the second-highest match is added into the match list provided the similarity > θ2.

After experimenting with different thresholds, θ1 = 0.55, and θ2 = 0.7 was found to produce the best F1 score with the most reasonable number of linked records.

Then, the ID pairs in the match list were written into ‘task1a.csv’.

param = []
a_match = []

for b_idx, b_item in buy_s.iterrows():
    
    scores = []
    
    for a_idx, a_item in abt_s.iterrows():
            
        similarity=0
        name_d = 0
        price_n = 0.5
        model_n = 0.2
        brand_n = 0
        a_name = (a_item['name'].replace("-",'').replace("/",'').replace("\'",'')).lower()
        a_name = nltk.word_tokenize(a_name)
        b_name = (b_item['name'].replace("-",'').replace("/",'').replace("\'",'')).lower()
        b_name = nltk.word_tokenize(b_name)                            
        name_d = textdistance.cosine.normalized_similarity(a_name, b_name)
        
        if pd.notna(a_item['price']) and pd.notna(b_item['price']):
            diff = abs(a_item['price'] - b_item['price'])
            if diff < 10:
                price_n = 1
            elif diff > 50:
                price_n = 0
        
        mod_a = a_name[-1]
        mod_b = b_name[-1]
        
        brand_a = a_name[0]
        brand_b = b_name[0]
        
        if ((mod_a == mod_b and (not mod_a.isalpha()) and (not mod_b.isalpha())) 
            or (not mod_a.isalpha() and mod_a in b_name) or (not mod_b.isalpha() and mod_b in a_name)) :
            model_n = 1
        
        if brand_a == brand_b:
            brand_n = 1
            
        similarity = 0.5*name_d + 0.2*model_n + 0.2*brand_n + 0.1*price_n
        
        scores.append([a_item['idABT'], similarity, a_item['name']])
        
        
    scores = sorted(scores, key=lambda x: x[1], reverse=True)
    if scores[0][1] > 0.55:
        if scores[0][0] not in a_match:
            param.append([scores[0][0], b_item['idBuy'], scores[0][2], b_item['name'], scores[0][1]])
        elif scores[0][0] in a_match and scores[1][1] > 0.7:
            param.append([scores[1][0], b_item['idBuy'], scores[1][2], b_item['name'], scores[1][1]])
            a_match.append(scores[1][0])
    
    if scores[0][1] > 0.7:
        a_match.append(scores[0][0])
    


task1a_csv=open("task1a.csv",'w')
writer1a = csv.writer(task1a_csv)
writer1a.writerow(['idABT','idBuy'])

for row in param:
    writer1a.writerow([row[0], row[1]])

Analysis of Naive Method

This product comparison algorithm yields a recall of 0.812 and precision of 0.840.

The number of true negatives was highest, meaning that the algorithm managed to differentiate between completely different products. However, there were some false positives, meaning that the similarity function failed to pick up some of the more nuanced differences in product names (e.g. same product name different models: “Panasonic Corded Phone - KXTS3282B”, “Panasonic KX-TS208W Corded Phone”).

There were also some false negatives, which indicates that the algorithm wasn’t able to recognise the same products with different product name details (e.g. same model but different model format: “GE Futura Indoor TV Antenna - Silver Finish - TV24746”, “Ge 24746 Futura(tm) Indoor Hdtv Antenna”).

Overall, the algorithm performed well on picking up most matches, but failed to pick up some details. The false negatives and positives were much lower compared to true positive and negatives.

To improve the performance, we could first examine attributes such as description, and calculate the similarity between description strings, assigning a default value if not exists, and account into our scoring function. We could also try and average the scores with other similarity measures that work well with strings of words such as Jaccard Similarity, to pick up more similarity/differences between strings. Also, we could use the recordlinkage library to assist with linking data.


Blocking

Script

Libraries Used

import pandas as pd
import csv

import nltk
from nltk.stem.porter import *
from nltk.stem import WordNetLemmatizer 
from nltk.corpus import stopwords
import re

Load CSV’s

task1ba_csv=open("abt_blocks.csv",'w')
writer1ba = csv.writer(task1ba_csv)
writer1ba.writerow(['block_key','product_id'])

task1bb_csv=open("buy_blocks.csv",'w')
writer1bb = csv.writer(task1bb_csv)
writer1bb.writerow(['block_key','product_id'])

buy_s=pd.read_csv('buy.csv',encoding = 'ISO-8859-1')
abt_s=pd.read_csv('abt.csv',encoding = 'ISO-8859-1')

The two datasets ‘abt.csv’ and. ‘buy.csv’ were opened and loaded into dataframes, then important features were extracted and cleaned using regular expressions.

buy_s=buy_s.loc[:,['idBuy','name','description','price','manufacturer']]
abt_s=abt_s.loc[:,['idABT','name','description','price']]

buy_s[['price']] = buy_s[['price']].replace('[\$,]','',regex=True).astype(float)
abt_s[['price']] = abt_s[['price']].replace('[\$,]','',regex=True).astype(float)

To place each item into blocks, a for loop is used. It first gets the product name, tokenises it using the nltk library, and removes punctuations and stop words from the word list, to avoid large, meaningless blocks.

Then, each word in the word list is lemmatised using WordNetLemmatizer from the nltk library, and a block is created for each word in the processed product name.

Before adding into the final list, some filtering works were also done to filter out potentially large blocks with generic product names or properties. Upon examining both datasets, colours and popular brand names were found to be creating unnecessary large blocks that will make the data linkage unproductive. Hence a small list including “sony”, “black” was created to help with filtering out the blocks.

Finally, we add the block key (which is the word) and the product ID into the csv file. The same process was done with both ‘abt.csv’ and. ‘buy.csv’. At the end of the code, each product was assigned to at least one block, with the block key being a component of its name.

Lemm = WordNetLemmatizer()
stopWords = set(stopwords.words('english'))

common_block = ['black','green','white','blue','yellow','red','purple','sony','panasonic','canon','digital','camera','lcd','system']

for a_idx, a_item in abt_s.iterrows():
    name = a_item['name']

    wordList = nltk.word_tokenize(name)
    wordList = [w.lower() for w in wordList]
    wordList = [word for word in wordList if word.isalpha()]

    filteredList = [w for w in wordList if not w in stopWords]
    
    for word in filteredList:
        stemWord = Lemm.lemmatize(word)
        if stemWord not in common_block:
            writer1ba.writerow([stemWord,a_item['idABT']])
            
for b_idx, b_item in buy_s.iterrows():
    name = b_item['name']

    wordList = nltk.word_tokenize(name)
    wordList = [w.lower() for w in wordList]
    wordList = [word for word in wordList if word.isalpha()]

    filteredList = [w for w in wordList if not w in stopWords]
    
    for word in filteredList:
        stemWord = Lemm.lemmatize(word)
        if stemWord not in common_block:
            writer1bb.writerow([stemWord,b_item['idBuy']])

Analysis of Blocking Method

This blocking method produced a pair completeness of 0.965 and reduction ratio of 0.931.

The time complexity in this method is O(m+n), with m, n being the size of each dataset. This means that all record is assigned to blocks in linear time, and the algorithm only goes through each product once.

Most matching records were put in the same block, producing high true positives (1059) and low false negatives (38), hence pair completeness is high. This was done by creating a large number of blocks, and by this most blocks having a low number of products assigned to it were able to be compared quickly and found matches. Most products that do not match with each other were able to be in different blocks, producing high true negatives.

There is still a significant number of false positives, of some product pairs with common generic properties in their name, these were placed into blocks with a rather high number of elements, such as ‘series’. However, the false positives (80392) are still much less than true negatives (1098963), hence when high true negatives added with low false negatives, the reduction ratio calculated was still rather high.

reduction ratio = 1 − (tp + fp)/n = (tn + fn)/n, n=fp+fn+tp+tn

To improve the performance, we can consider creating blocks with multiple n-words, price, or a combination of both. This should reduce the number of pairs in the blocks, thus speeding up the comparison. We could also examine and add the description to the blocks in addition to name and price, if exists, to improve accuracy.