How to import modules from parent sub-directory
Asked Answered
G

4

6

I want to import a class from a python file in a directory in the parent directory. This is my current folder structure:

│   run_simulation_from_parent.py
│   __init__.py
│
├───.vscode
│       launch.json
│       settings.json
│
├───mod
│   │   mods.py
│   │   __init__.py
│   │
│   └───__pycache__
│           __init__.cpython-37.pyc
│
├───sim
│   │   run_simulation.py
│   │   __init__.py
│   │
│   └───__pycache__
│           __init__.cpython-37.pyc
│
└───__pycache__
        __init__.cpython-37.pyc

The file mod/mods.py contains the following class:

class Objective:
    """Objective function class"""

    def __init__(self, x):
        self.x = x

The file sim/run_simulation.py contains:

from mod.mods import Objective

x = 5
obj = Objective(x)

When I try to run this I get the following error:

  File "sim/run_simulation.py", line 1, in <module>
    from mod.mods import Objective
ModuleNotFoundError: No module named 'mod'

In visual studio code it does autofill when I start typing mod.mods and import Objective

When I run run_simulation_from_parent.py with the following content I have no problems:

from mod.mods import Objective

x = 5
obj = Objective(x)

print(obj.x)

How can I do this from the directory sim? I already tried the following:

  1. Use from ..mod.mods import Objective to run_simulation.py
  2. Use init.py files with the following content: import os, sys sys.path.append(os.path.dirname(os.path.realpath(__file__)))

  3. Without the __init__.py files

Edit: I run the file from visual studio code where I start in the parent directory. I also tried from the command line in windows from the sim folder where I used

python run_simulation.py
Guglielma answered 28/6, 2019 at 12:3 Comment(3)
Possible duplicate of Import a file from a subdirectory?Frankel
@Guglielma can you tell the way you are using the project, ie from where you are running the run_simulations.py file ? means how you are running it in terminalLupita
I run the file from visual studio code where I start in the parent directory.Guglielma
B
6

The way I deal with imports inside a project is to install the project in editable mode. This way, all files will be able to locate each other, always starting from your project root directory.

In order to do this, follow these steps:

1) write a setup.py file and add it to your project root folder - it doesn't need much info at all:

# setup.py
from setuptools import setup, find_packages

setup(name='MyPackageName', version='1.0.0', packages=find_packages())

2) install your package in editable mode (ideally from a virtual environment). From a terminal in your project folder, write

$ pip install -e .

Note the dot - this means "install the package from the current directory in editable mode".

3) your files inside the project are now able to locate each other, always starting from the project root. To import Objective, for example, you write:

from mod.mods import Objective

This will be true to import Objective for any file, no matter where it is located in the project structure.

Like I said, you should use a virtual environment for this, so that pip does not install your package to your main Python installation (which could be messy if your project has many dependencies).

My favorite tool for this is pipenv. When using it, replace the terminal command with

$ pipenv install -e .

So that your project gets added to the Pipfile.

Birck answered 28/6, 2019 at 12:15 Comment(1)
Thanks for you elaborate explanation! I tried what you said and used 'setup(name='Objective', version='1.0.0', packages=find_packages())' I ran the pip command and a folder called Objective.egg-info is created. It doesn't solve my problem though..Guglielma
W
1

Use from ..mod.mods import Objective and keep all the __init__.py files.

Weakling answered 28/6, 2019 at 12:8 Comment(1)
My bad, there was a typo in the OP. I already tried ..mod.modsGuglielma
G
1

I did find one solution by adding it to the system path:

import os
import os.path
import sys

sys.path.append(os.getcwd() + "\\mod")

from mods import Objective

x = 5
obj = Objective(x)

print(obj.x)

This works but I cannot imagine that this is really how it is intended..

Edit: The following is a bit more general:

import os
import sys

sys.path.append(os.getcwd())

from mod.mods import Objective

x = 5
obj = Objective(x)

print(obj.x)

Something tells me that this could be done easier but I don't know how..

Guglielma answered 28/6, 2019 at 12:46 Comment(0)
S
1

There is one specific suggestion for those who, like the author, use VS Code. In launch.json you can add env dictionary with "PYTHONPATH":"${workspaceFolder}" key-value pair.

e.g.:

{
    "version": "0.2.0",
    "configurations": [
        {
            "name": "Python: Current File",
            "type": "python",
            "request": "launch",
            "program": "${file}",
            "console": "integratedTerminal",
            "justMyCode": true,
            "env": {
                "PYTHONPATH": "${workspaceFolder}"}
        }
    ]
}
Sternberg answered 1/4, 2024 at 21:3 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.