{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9dd25478-e9e7-4b50-8987-af6a60fa96ec",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import tensorflow as tf\n",
    "from nltk.corpus import wordnet\n",
    "import torch\n",
    "import pickle\n",
    "import gensim\n",
    "import warnings\n",
    "from itertools import product\n",
    "warnings.filterwarnings('ignore')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "615dbfa0-207b-43b0-b5be-4ded5d96d063",
   "metadata": {},
   "outputs": [],
   "source": [
    "#get sets of all parts of speech from wordnet\n",
    "\n",
    "allNoun = set([synset.lemmas()[0].name().lower() for synset in list(wordnet.all_synsets('n')) if\n",
    "              synset.lexname().split(\".\")[0]==\"noun\"])\n",
    "\n",
    "allVerb = set([synset.lemmas()[0].name().lower() for synset in list(wordnet.all_synsets('v')) if\n",
    "              synset.lexname().split(\".\")[0]==\"verb\"])\n",
    "\n",
    "allAdje = set([synset.lemmas()[0].name().lower() for synset in \n",
    "               (list(wordnet.all_synsets('a')) + list(wordnet.all_synsets('s'))) if\n",
    "              synset.lexname().split(\".\")[0]==\"adj\"])\n",
    "\n",
    "allAdve = set([synset.lemmas()[0].name().lower() for synset in list(wordnet.all_synsets('r')) if\n",
    "              synset.lexname().split(\".\")[0]==\"adv\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a730162f-81df-4349-bdca-1557e044e72f",
   "metadata": {},
   "outputs": [],
   "source": [
    "#import remaining parts of speech from other sources\n",
    "\n",
    "#list of pronouns from https://www.thoughtco.com/\n",
    "pronouns = ['anybody', 'anyone', 'anything', 'aught', \n",
    "            'either', 'enough', 'everybody', 'everyone', 'everything', 'he', 'her', 'hers', \n",
    "            'herself', 'him', 'himself', 'his', 'i', 'idem', 'it', 'its', 'itself', 'many', 'me', \n",
    "            'mine', 'most', 'my', 'myself', 'naught', 'neither', 'nobody', 'none', 'nothing', 'nought', \n",
    "            'one', 'other', 'others', 'ought', 'our', 'ours', 'ourself', 'ourselves', 'several', 'she', \n",
    "            'some', 'somebody', 'someone', 'something', 'somewhat', 'such', 'suchlike', 'that', 'thee', \n",
    "            'their', 'theirs', 'theirself', 'theirselves', 'them', 'themself', 'themselves', 'there', \n",
    "            'these', 'they', 'thine', 'this', 'those', 'thou', 'thy', 'thyself', 'us', 'we', 'what', \n",
    "            'whatever', 'whatnot', 'whatsoever', 'whence', 'where', 'whereby', 'wherefrom', 'wherein', \n",
    "            'whereinto', 'whereof', 'whereon', 'wherever', 'wheresoever', 'whereto', 'whereunto', \n",
    "            'wherewith', 'wherewithal', 'whether', 'which', 'whichever', 'whichsoever', 'who', \n",
    "            'whoever', 'whom', 'whomever', 'whomso', 'whomsoever', 'whose', 'whosever', 'whosesoever', \n",
    "            'whoso', 'whosoever', 'ye', 'yon', 'yonder', 'you', 'your', 'yours', 'yourself', 'yourselves']\n",
    "\n",
    "#list of all possible conjunctions from https://englishgrammarhere.com/\n",
    "\n",
    "conjunctions = ['later', 'whereas', 'besides', 'however', 'after', 'least', 'whomever', 'because', \n",
    "                'accordingly', 'nonetheless', 'so', 'third', 'actually', 'still', 'thus', 'further', \n",
    "                'too', 'lastly', 'whoever', 'thereafter', 'soon', 'fourth', 'finally', 'presently', \n",
    "                'and', 'what', 'before', 'briefly', 'afterwards', 'hence', 'then', 'since', \n",
    "                'consequently', 'ultimately', 'yet', 'whatever', 'also', 'last', 'first', 'but', \n",
    "                'meanwhile', 'when', 'conversely', 'furthermore', 'nevertheless', 'another', \n",
    "                'moreover', 'or', 'therefore', 'now', 'nor', 'subsequently', 'similarly', 'gradually', \n",
    "                'while', 'second', 'next']\n",
    "\n",
    "#list of all prepositions collected from https://7esl.com/list-of-prepositions/\n",
    "\n",
    "prepositions = ['above', 'whether', 'besides', 'in', 'throughout', 'between', 'astride', 'regarding', \n",
    "                'counting', 'saving', 'below', 'outside', 'through', 'off', 'including', 'but', 're', \n",
    "                'via', 'except', 'among', 'on', 'along', 'over', 'concerning', 'than', 'far', \n",
    "                'excepting', 'onto', 'underneath', 'given', 'following', 'amid', 'without', 'pending', \n",
    "                'since', 'near', 'as', 'of', 'about', 'bar', 'until', 'unlike', 'by', 'round', 'versus', \n",
    "                'anti', 'minus', 'from', 'beyond', 'toward', 'away', 'considering', 'save', 'behind',\n",
    "                'past', 'despite', 'beside', 'cum', 'opposite', 'aboard', 'till', 'circa', 'to', 'plus',\n",
    "                'out', 'ahead', 'per', 'atop', 'barring', 'inside', 'into', 'beneath', 'excluding', \n",
    "                'up', 'pro', 'down', 'after', 'like', 'alongside', 'against', 'ago', 'at', 'during', \n",
    "                'less', 'with', 'notwithstanding', 'for', 'before', 'under', 'within', 'around', \n",
    "                'across', 'worth', 'upon', 'gone']\n",
    "\n",
    "interjections = ['aha', 'ahem', 'ahh', 'ahoy', 'alas', 'arg', 'aw', 'bam', 'bingo', 'blah', 'boo', \n",
    "                 'bravo', 'brrr', 'cheers', 'congratulations', 'dang', 'drat', 'darn', 'duh', 'eek', \n",
    "                 'eh', 'encore', 'eureka', 'fiddlesticks', 'gadzooks', 'gee', 'gee whiz', 'golly', \n",
    "                 'goodbye', 'goodness', 'good grief', 'gosh', 'hallelujah', 'hello', 'hey', \n",
    "                 'hmm', 'holy buckets', 'holy cow', 'holy smokes', 'hot dog', 'huh', 'humph', 'hurray', \n",
    "                 'oh', 'oh dear', 'oh my', 'oh well', 'oops', 'ouch', 'ow', 'phew', 'phooey', 'pooh', \n",
    "                 'pow', 'rats', 'shh', 'shoo', 'thanks', 'there', 'ugh', 'wahoo', 'well', 'whoa', \n",
    "                 'whoops', 'wow', 'yeah', 'yes', 'yikes', 'yippee', 'yo', 'yuck']\n",
    "\n",
    "allPron = set(pronouns)\n",
    "allConj = set(conjunctions)\n",
    "allPrep = set(prepositions)\n",
    "allInte = set(interjections)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "771808c3-072f-4a32-bd3d-c3fa8c36a491",
   "metadata": {},
   "outputs": [],
   "source": [
    "allPos = [allNoun, allVerb, allAdje, allAdve, allPron, allConj, allPrep, allInte]\n",
    "\n",
    "#remove all words that belong to multiple parts of speeches\n",
    "\n",
    "onlyNoun = list(allNoun - set().union(*[i for i in allPos if i!=allNoun]))\n",
    "onlyVerb = list(allVerb - set().union(*[i for i in allPos if i!=allVerb]))\n",
    "onlyAdje = list(allAdje - set().union(*[i for i in allPos if i!=allAdje]))\n",
    "onlyAdve = list(allAdve - set().union(*[i for i in allPos if i!=allAdve]))\n",
    "onlyPron = list(allPron - set().union(*[i for i in allPos if i!=allPron]))\n",
    "onlyConj = list(allConj - set().union(*[i for i in allPos if i!=allConj]))\n",
    "onlyPrep = list(allPrep - set().union(*[i for i in allPos if i!=allPrep]))\n",
    "onlyInte = list(allInte - set().union(*[i for i in allPos if i!=allInte]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fae7852e-3a72-4b00-ad50-77dda3908714",
   "metadata": {},
   "outputs": [],
   "source": [
    "#filter all words to remove non-alphabets\n",
    "\n",
    "def preprocessWords(pos):\n",
    "    for ind, word in enumerate(pos):\n",
    "        if not word.isalpha():    \n",
    "            if \"-\" in word: pos[ind] = word.split(\"-\")\n",
    "            elif \"_\" in word: pos[ind] = word.split(\"_\")\n",
    "            elif \" \" in word: pos[ind] = word.split()\n",
    "        else:\n",
    "            pos[ind] = [word]\n",
    "    return pos\n",
    "                \n",
    "onlyNoun = preprocessWords(onlyNoun)\n",
    "onlyVerb = preprocessWords(onlyVerb)\n",
    "onlyAdje = preprocessWords(onlyAdje)\n",
    "onlyAdve = preprocessWords(onlyAdve)\n",
    "onlyPron = preprocessWords(onlyPron)\n",
    "onlyConj = preprocessWords(onlyConj)\n",
    "onlyPrep = preprocessWords(onlyPrep)\n",
    "onlyInte = preprocessWords(onlyInte)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0295c1db-5196-4159-bfa0-bbbeb52640d5",
   "metadata": {},
   "outputs": [],
   "source": [
    "#impute values to ensure all part of speech have equal number of candidates\n",
    "\n",
    "def imputer(wordList):\n",
    "    impValue = int(100000/len(wordList))+1\n",
    "    wordList = wordList*impValue\n",
    "    return wordList[1:100001]\n",
    "\n",
    "onlyNounImputed = imputer(onlyNoun)\n",
    "onlyVerbImputed = imputer(onlyVerb)\n",
    "onlyAdjeImputed = imputer(onlyAdje)\n",
    "onlyAdveImputed = imputer(onlyAdve)\n",
    "onlyPronImputed = imputer(onlyPron)\n",
    "onlyConjImputed = imputer(onlyConj)\n",
    "onlyPrepImputed = imputer(onlyPrep)\n",
    "onlyInteImputed = imputer(onlyInte)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8597f3c7-f9de-4589-a16e-0141a4d4df84",
   "metadata": {},
   "outputs": [],
   "source": [
    "#load glove using gensim\n",
    "\n",
    "glove_path = \"/glove840b300dtxt/glove.840B.300d.txt\"\n",
    "glove = gensim.models.keyedvectors.load_word2vec_format(fname=glove_path, binary=False, no_header=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "67b843c5-f501-4943-ae67-37b33d21aa02",
   "metadata": {},
   "outputs": [],
   "source": [
    "#collect glove embeddings for all candidates\n",
    "\n",
    "pos = [onlyNounImputed, onlyVerbImputed, onlyAdjeImputed, onlyAdveImputed, onlyPronImputed, \n",
    "       onlyConjImputed, onlyPrepImputed, onlyInteImputed]\n",
    "gloveNoun, gloveVerb, gloveAdje, gloveAdve, glovePron, gloveConj, glovePrep, gloveInte = [], [], [], [], [], [], [], []\n",
    "gloveEmb = [gloveNoun, gloveVerb, gloveAdje, gloveAdve, glovePron, gloveConj, glovePrep, gloveInte]\n",
    "\n",
    "for i in range(len(pos)):\n",
    "    print(i)\n",
    "    for words in pos[i]:\n",
    "        if len(words) == 1:\n",
    "            word = words[0]\n",
    "            if glove.__contains__(word):\n",
    "                gloveEmb[i].append(glove.__getitem__(word))\n",
    "        else:\n",
    "            if all([glove.__contains__(word) for word in words]):\n",
    "                combinedEmb = np.array([glove.__getitem__(word) for word in words])\n",
    "                gloveEmb[i].append(np.average(combinedEmb, axis=0))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e342f27a-3d11-4e3f-b090-a3554d22f0c8",
   "metadata": {},
   "outputs": [],
   "source": [
    "#generate syntactic glove absolutes using the syntactic candidates\n",
    "\n",
    "def generateGloveAbsolutes(posGloveList):\n",
    "    absGlovePos = np.average(posGloveList, axis=0)\n",
    "    return absGlovePos.tolist()\n",
    "\n",
    "absGloveNoun = generateGloveAbsolutes(gloveNoun)\n",
    "absGloveVerb = generateGloveAbsolutes(gloveVerb)\n",
    "absGloveAdje = generateGloveAbsolutes(gloveAdje)\n",
    "absGloveAdve = generateGloveAbsolutes(gloveAdve)\n",
    "absGlovePron = generateGloveAbsolutes(glovePron)\n",
    "absGloveConj = generateGloveAbsolutes(gloveConj)\n",
    "absGlovePrep = generateGloveAbsolutes(glovePrep)\n",
    "absGloveInte = generateGloveAbsolutes(gloveInte)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "45c3bc2d-274b-446a-8b28-841dd84c8daa",
   "metadata": {},
   "outputs": [],
   "source": [
    "gloveDict = {\n",
    "            \"noun\": absGloveNoun,\n",
    "            \"verb\": absGloveVerb,\n",
    "            \"adje\": absGloveAdje,\n",
    "            \"adve\": absGloveAdve,\n",
    "            \"pron\": absGlovePron,\n",
    "            \"conj\": absGloveConj,\n",
    "            \"prep\": absGlovePrep,\n",
    "            \"inte\": absGloveInte,\n",
    "             }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f1279245-514b-4614-baa8-d16b2b9decc2",
   "metadata": {},
   "outputs": [],
   "source": [
    "#load word2vec\n",
    "\n",
    "file_path = \"GoogleNews-vectors-negative300.bin\"\n",
    "word2vec = gensim.models.keyedvectors.load_word2vec_format(fname=file_path, binary=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "864e8167-8a4e-41ff-a618-c52d5ce4784b",
   "metadata": {},
   "outputs": [],
   "source": [
    "#get word2vec embeddings for all candidates\n",
    "\n",
    "pos = [onlyNounImputed, onlyVerbImputed, onlyAdjeImputed, onlyAdveImputed, onlyPronImputed, \n",
    "       onlyConjImputed, onlyPrepImputed, onlyInteImputed]\n",
    "w2vNoun, w2vVerb, w2vAdje, w2vAdve, w2vPron, w2vConj, w2vPrep, w2vInte = [], [], [], [], [], [], [], []\n",
    "w2vEmb = [w2vNoun, w2vVerb, w2vAdje, w2vAdve, w2vPron, w2vConj, w2vPrep, w2vInte]\n",
    "\n",
    "for i in range(len(pos)):\n",
    "    print(i)\n",
    "    for words in pos[i]:\n",
    "        if len(words) == 1:\n",
    "            word = words[0]\n",
    "            if word2vec.__contains__(word):\n",
    "                w2vEmb[i].append(word2vec.__getitem__(word))\n",
    "        else:\n",
    "            if all([word2vec.__contains__(word) for word in words]):\n",
    "                combinedEmb = np.array([word2vec.__getitem__(word) for word in words])\n",
    "                w2vEmb[i].append(np.average(combinedEmb, axis=0))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "eedb1f85-ad97-44bf-ab1f-68857bb04071",
   "metadata": {},
   "outputs": [],
   "source": [
    "#generate syntactic word2vec absolutes using the syntactic candidates\n",
    "\n",
    "def generateW2VAbsolutes(posW2VList):\n",
    "    absW2VPos = np.average(posW2VList, axis=0)\n",
    "    return absW2VPos.tolist()\n",
    "\n",
    "absW2VNoun = generateW2VAbsolutes(w2vNoun)\n",
    "absW2VVerb = generateW2VAbsolutes(w2vVerb)\n",
    "absW2VAdje = generateW2VAbsolutes(w2vAdje)\n",
    "absW2VAdve = generateW2VAbsolutes(w2vAdve)\n",
    "absW2VPron = generateW2VAbsolutes(w2vPron)\n",
    "absW2VConj = generateW2VAbsolutes(w2vConj)\n",
    "absW2VPrep = generateW2VAbsolutes(w2vPrep)\n",
    "absW2VInte = generateW2VAbsolutes(w2vInte)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3d31a52b-5c89-49a8-9841-0d21a7825584",
   "metadata": {},
   "outputs": [],
   "source": [
    "w2vDict= {\"noun\": absW2VNoun,\n",
    "             \"verb\": absW2VVerb,\n",
    "             \"adje\": absW2VAdje,\n",
    "             \"adve\": absW2VAdve,\n",
    "             \"pron\": absW2VPron,\n",
    "             \"conj\": absW2VConj,\n",
    "             \"prep\": absW2VPrep,\n",
    "             \"inte\": absW2VInte,\n",
    "             }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "edf9fba9-3d08-48a5-ab31-ac55aaabeeba",
   "metadata": {},
   "outputs": [],
   "source": [
    "absPosEmbListW2V = [w2vDict[i] for i in w2vDict.keys()]\n",
    "absPosEmbListGlove = [gloveDict[i] for i in gloveDict.keys()]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c99d9437-51e2-4734-b9c0-501a6ced7532",
   "metadata": {},
   "outputs": [],
   "source": [
    "#load spider chart\n",
    "def loadSpiderNon(final, word):\n",
    "    import plotly.express as px\n",
    "    import pandas as pd\n",
    "    df = pd.DataFrame(dict(\n",
    "        r=final,\n",
    "        theta=['noun','verb','adjective','adverb','pronoun','conjunction','preposition','interjection']))\n",
    "    fig = px.line_polar(df, r='r', theta='theta', line_close=True)\n",
    "    fig.update_layout(title_text=word, title_x=0.5)\n",
    "    fig.update_traces(fill='toself')\n",
    "    fig.show()\n",
    "\n",
    "def generateResultNon(word, flag, interpret=False):\n",
    "    if flag == \"glove\":\n",
    "        if glove.__contains__(word):\n",
    "            testEmbedding = glove.__getitem__(word)\n",
    "        else:\n",
    "            testEmbedding = 0\n",
    "        direReal = np.array(absPosEmbListGlove, dtype=\"float32\")\n",
    "        \n",
    "    elif flag == \"word2vec\":\n",
    "        if word2vec.__contains__(word):\n",
    "            testEmbedding = word2vec.__getitem__(word)\n",
    "        else:\n",
    "            testEmbedding = 0\n",
    "        direReal = np.array(absPosEmbListW2V, dtype=\"float32\")\n",
    "    if testEmbedding is 0:\n",
    "        final = 0 # implies that the word is not in the models vocabulary\n",
    "    else:\n",
    "        direReal = direReal.transpose()\n",
    "        direRealInv = np.linalg.pinv(direReal)\n",
    "        subEmbdReal = np.matmul(direRealInv, testEmbedding) \n",
    "        \n",
    "        final = subEmbdReal #absolute syntactic representations\n",
    "#         final = ((subEmbdReal - min(subEmbdReal)) / (max(subEmbdReal) - min(subEmbdReal)) +1)/2 #interpretable syntactic representations\n",
    "#         final = subEmbdReal / np.linalg.norm(subEmbdReal) #L2 syntactic representations\n",
    "        \n",
    "        if interpret:\n",
    "            allFinal = final.tolist()\n",
    "            loadSpiderNon(allFinal, word)\n",
    "    return final"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "85ae4ac3-74c9-4f13-b01c-193e2834382d",
   "metadata": {},
   "outputs": [],
   "source": [
    "#create a model for the syntactic representation\n",
    "\n",
    "file = open(\"glovePos.bin\", \"wb\")\n",
    "file.write(str.encode(str(len(neededWords)) + \" \" + \"8\\n\"))\n",
    "\n",
    "for word in neededWords:\n",
    "    file.write(str.encode(word+' '))\n",
    "    file.write(generateResultNon(word, \"glove\"))\n",
    "    file.write(str.encode(\"\\n\"))\n",
    "file.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "afbc728f-e468-4fdb-80c3-f19919d5fd86",
   "metadata": {},
   "outputs": [],
   "source": [
    "syntacticModel = gensim.models.keyedvectors.load_word2vec_format(\"glovePos_unNorm.bin\", binary=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0a04fabd-ab85-4a7a-8908-e407a2e6f2f9",
   "metadata": {},
   "outputs": [],
   "source": [
    "#derive the Hierarchical Overcomplete Vector\n",
    "\n",
    "temp = open(\"glove_chained_2400.bin\", \"wb\")\n",
    "cnt = len([i for i in syntModel.key_to_index if i in glove.key_to_index])\n",
    "temp.write(str.encode(str(cnt) + ' '+ \"2400\\n\"))\n",
    "\n",
    "for i in glove.key_to_index:\n",
    "    if i in syntModel:\n",
    "        mp = syntModel[i]\n",
    "        vector = glove[i]\n",
    "        repVectorList = list(product(mp, vector))\n",
    "        repVectorMul = np.array([i[0]*i[1] for i in repVectorList])\n",
    "        repVector = (repVectorMul - min(repVectorMul)) / (max(repVectorMul) - min(repVectorMul))\n",
    "        repVector = repVectorMul.astype(\"float32\")\n",
    "        temp.write(str.encode(i+\" \")) \n",
    "        temp.write(repVector) \n",
    "        temp.write(str.encode(\"\\n\"))\n",
    "temp.close()\n",
    "del temp"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4d3610dc-ebb1-4194-b903-c5b20e1ce5cc",
   "metadata": {},
   "outputs": [],
   "source": [
    "#derive the Hierarchical Weighted Vector\n",
    "\n",
    "temp = open(\"glove_chained_300_unNorm.bin\", \"wb\")\n",
    "cnt = len([i for i in syntModel.key_to_index if i in glove.key_to_index])\n",
    "temp.write(str.encode(str(cnt) + ' '+ \"300\\n\"))\n",
    "\n",
    "for i in glove.key_to_index:\n",
    "    if i in syntModel:\n",
    "        repVector2 = [0]*300\n",
    "        mp = syntModel[i]\n",
    "        vector = glove[i]\n",
    "        repVector2 = [sum([j*vector[i] for j in mp]) for i in range(len(vector))]\n",
    "        repVector2 = np.array(repVector2, dtype=\"float32\")\n",
    "        temp.write(str.encode(i+\" \")) \n",
    "        temp.write(repVector2) \n",
    "        temp.write(str.encode(\"\\n\"))\n",
    "temp.close()\n",
    "del temp"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1116db1d-ac84-4eeb-9753-3f94fa06f34e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "07e3199e-514b-4114-95cc-2a1389986dda",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9914aa29-21af-4f06-ab1a-48e9ae03285e",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
