Python bulk import animals

Morning/ evening/ whatever time is is for you folk.

I am working on a script that can be executed from any machine to import an animal and make sure the breed/species is also set. does anyone have a working script i can look over to see where i am missing things?

Currently using this script.

from farmOS import farmOS
from datetime import datetime
import csv


hostname = "farming.*****.*******"
username = *****************
password = **********************

# Maintain an external state of the token.
current_token = None

# Callback function to save new tokens.
def token_updater(new_token):
    print(f"Got a new token! {new_token}")
    # Update state.
    current_token = new_token

# Create the client.
farm_client = farmOS(
    hostname=hostname,
    token_updater=token_updater, # Provide the token updater callback.
)

# Authorize the client (initial or refresh token):
if not current_token:
    current_token = farm_client.authorize(username, password, scope="farm_manager")
else:
    # Use existing token if available (optional refresh logic)
    pass


# Create the animal
animal_data = {
    "attributes": {
        "name": "My Great Planting",
    },
}

response = farm_client.asset.send('animal', animal_data)

but i am getting an error at the last line saying
“”“422 Client Error: Unprocessable Content for url”“”

Any assistance would be greatly helpful.

1 Like

I haven’t done this in Python, but have using Node-Red and just using JS.

Is there a message field in the response with a more detailed description of what’s failing, it may give you a better idea of what the issue is.

1 Like

Did you check out my tutorial blog post on python scripting with farmOS?

1 Like

Welcome to the forum @mattman0123!

Sounds like you’ve already identified the issue! The “Animal Type” (animal_type) field is required on animal assets (aka “Species/breed”). It belongs in the relationships of the asset. If you create an animal through the farmOS UI you can look at it in the API to see how animal_type needs to be formatted. It needs to reference the UUID of a taxonomy_term in the animal_type vocabulary.

You can fetch a list of animal_type terms from the API to find the UUID of the term you want. If the term does not exist, you will need to create it first before you can reference it on the animal asset.

Once you have the UUID, I think your code will look something like this:

# Create the animal
animal_data = {
    "attributes": {
        "name": "My Great Animal",
    },
    "relationships": {
        "animal_type": {
            "data": {
                "id": "5d9cce61-032f-4dba-b67c-999429505a12",
            },
        },
    },
}
1 Like

good to know this is a required. I will iterate through the animal_type and see if i can figure out how to add that into the import.

1 Like

That’s already covered in the tutorial blog post I linked:

I expect it shouldn’t take much to adapt it to your needs.

3 Likes

This is perfect looking at this now.
ill update once i am done.

2 Likes

SOLUTION SO FAR:

from farmOS import farmOS
from datetime import datetime
import pandas as pd
import csv


hostname = "farming.****************.cloud"
username = "*************"
password = "*************"

# Maintain an external state of the token.
current_token = None

# Callback function to save new tokens.
def token_updater(new_token):
    print(f"Got a new token! {new_token}")
    # Update state.
    current_token = new_token

# Create the client.
farm_client = farmOS(
    hostname=hostname,
    token_updater=token_updater, # Provide the token updater callback.
)

# Authorize the client (initial or refresh token):
if not current_token:
    current_token = farm_client.authorize(username, password, scope="farm_manager")
else:
    # Use existing token if available (optional refresh logic)
    pass

def create_find_animal_type(data):
    animal_type_search = farm_client.term.iterate(
        'animal_type',
        params=farm_client.filter('name', data),
    )
    animal_type = next(iter(animal_type_search), None)

    # If the animal type does not already exist, create it
    if animal_type is None:
        term_create_response = farm_client.term.send(
            'animal_type',
            {"attributes": {"name": data}}
        )
        animal_type = term_create_response["data"]
    return animal_type


with open(r"csv_asset--animal.csv", newline='') as csvfile:
    csv_reader = csv.DictReader(csvfile)
    for animal in csv_reader:
        animal_dob = datetime.strptime(animal['birthdate'], "%Y-%m-%d")
        animal_type = create_find_animal_type(animal["animal type"])
        # Create the animal
        animal_create_response = farm_client.asset.send('animal', {
            "attributes": {
                "name": animal['name'],
                "sex": animal['sex'],
                "birthdate": animal_dob.strftime('%Y-%m-%dT%H:%M:%S+00:00'),
                "is fixed": animal["is fixed"],
            },
            "relationships": {
                "animal_type": {
                    # Make each animal a animal_type
                    "data": {
                        "type": animal_type['type'],
                        "id": animal_type['id'],
                    },
                },
            },
        })

        print("Created {!r}: {}/asset/{}".format(
            animal['name'],
            farm_client.session.hostname,
            animal_create_response['data']['attributes']['drupal_internal__id'],
        ))

this worked great for importing 100 hens into my coop. Then i manually moved them to where they needed to be. Also created the animal_type as required.

CSV was formatted as follows:

name,is fixed, animal type, birthdate, sex

2 Likes

@mattman0123 Cool! Glad you got it working!

Curious… is there a reason you’re allowing is_fixed to be changed? For animal assets that should generally be set to FALSE all the time, unless you’re doing something intentionally?

For more info on is_fixed, see these data model docs: Location | farmOS

It was just included in the default CSV I created when I accidentally thought it was for castrated or not.

Next CSV may remove that. I might adjust the script to auto bring in anything from the CSV that is not empty and auto create the payload a little nicer.

1 Like

:+1: Yea it’s not the most intuitive name.

On a related note, we will probably be changing is_castrated to is_sterile in farmOS 4.x, so be aware of that when the time comes to upgrade: https://www.drupal.org/project/farm/issues/3494784

added all the script to github, added more details and validation for the CSV.
For some reason my instance can not use the CSV import properly. so i built this up. This is also a way for me to automate some of the animal sales we conduct.

2 Likes

Thanks for sharing @mattman0123! I’m sure others will find this useful in the future.

I wonder if it’s a CSV upload issue? Are you able to upload other files, like images/files on assets/logs? Happy to help debug this if you want… maybe in a separate topic or chat.