The Clarifai API has the ability not only to learn concepts from images and videos, but from text as well.
In this walkthrough, you'll learn how to create and use a custom text model, learn from your own text data using the power of the Clarifai's base Text model, and predict on new text examples.
The examples below map directly to any of our other gRPC clients.
The walkthrough assumes you have already created your Clarifai's user account and the Personal Access Token. Also, first set up the gRPC Python client together with the initial code, see Client Installation Instructions.
For debugging purposes, each response returned by a method call can be printed to the console, and its entire data and structure will be shown verbosely.
Create a new application
We'll first create a new application to which we'll later be posting our training inputs. The default application's workflow that we we'll be selecting is going to be "Text". By selecting this base workflow, all custom models that we'll later be creating inside this application will be based of the Clarifai's default Text model.
For authentication, we'll for now use the Personal Access Token, which is able to do operations such as creating a new application and afterward, a new API key.
pat_metadata = (('authorization','Key {YOUR_PERSONAL_ACCESS_TOKEN}'),)post_apps_response = stub.PostApps( service_pb2.PostAppsRequest( apps=[ resources_pb2.App( id="my-text-to-concept-app", default_workflow_id="Text" ) ] ), metadata=pat_metadata)# You may print the response to see what the structure and the data of the response is.# print(post_apps_response)user_id = post_apps_response.apps[0].user_idif post_apps_response.status.code != status_code_pb2.SUCCESS:raiseException("Failed response, status: "+str(post_apps_response.status))
Create a new API key
API keys are tied to specific applications. Now that we have an application, let's create its API key. Since the API key will only to be able to operate on this application, it reduces the risk if the key is lost/stolen, and it avoids us specifying which application should be used in every method call (which is what we'd have to do if we kept using the PAT).
See at the bottom of the code example that we put the API key into a metadata tuple. We'll be using this metadata (and the API key) from now on for all the code examples that follow.
We'll now add several text inputs that we will later use as a training data in our custom model. The idea is that we'll create a model which can differentiate between positive and negative sentences (in a grammatical sense). We'll mark each input with one of the two concepts: positive or negative.
The text can be added either directly (it's called raw), or from a URL.
positive_raw_texts = ["Marie is a published author.","In three years, everyone will be happy.","Nora Roberts is the most prolific romance writer the world has ever known.","She has written more than 225 books.","If you walk into Knoxville, you'll find a shop named Rala.","There are more than 850 miles of hiking trails in the Great Smoky Mountains.","Harrison Ford is 6'1\".","According to Reader's Digest, in the original script of Return of The Jedi, Han Solo died.","Kate travels to Doolin, Ireland every year for a writers' conference.","Fort Stevens was decommissioned by the United States military in 1947.",]negative_text_urls = ["https://samples.clarifai.com/negative_sentence_1.txt","https://samples.clarifai.com/negative_sentence_2.txt","https://samples.clarifai.com/negative_sentence_3.txt","https://samples.clarifai.com/negative_sentence_4.txt","https://samples.clarifai.com/negative_sentence_5.txt","https://samples.clarifai.com/negative_sentence_6.txt","https://samples.clarifai.com/negative_sentence_7.txt","https://samples.clarifai.com/negative_sentence_8.txt","https://samples.clarifai.com/negative_sentence_9.txt","https://samples.clarifai.com/negative_sentence_10.txt",]post_inputs_response = stub.PostInputs( service_pb2.PostInputsRequest( inputs=[ resources_pb2.Input( data=resources_pb2.Data( text=resources_pb2.Text(raw=raw_text), concepts=[resources_pb2.Concept(id="positive", value=1)] ) )for raw_text in positive_raw_texts ] + [ resources_pb2.Input( data=resources_pb2.Data( text=resources_pb2.Text( url=text_url, allow_duplicate_url=True ), concepts=[resources_pb2.Concept(id="negative", value=1)] ) )for text_url in negative_text_urls ] ), metadata=api_key_metadata)if post_inputs_response.status.code != status_code_pb2.SUCCESS:raiseException("Failed response, status: "+str(post_inputs_response.status))
Wait for inputs to download
Let's now wait for all the inputs to download.
whileTrue: list_inputs_response = stub.ListInputs( service_pb2.ListInputsRequest(page=1, per_page=100), metadata=api_key_metadata )if list_inputs_response.status.code != status_code_pb2.SUCCESS:raiseException("Failed response, status: "+str(list_inputs_response.status))for the_input in list_inputs_response.inputs: input_status_code = the_input.status.codeif input_status_code == status_code_pb2.INPUT_DOWNLOAD_SUCCESS:continueelif input_status_code in (status_code_pb2.INPUT_DOWNLOAD_PENDING, status_code_pb2.INPUT_DOWNLOAD_IN_PROGRESS):print("Not all inputs have been downloaded yet. Checking again shortly.")breakelse: error_message = (str(input_status_code)+" "+ the_input.status.description +" "+ the_input.status.details )raiseException(f"Expected inputs to download, but got {error_message}. Full response: {list_inputs_response}" )else:# Once all inputs have been successfully downloaded, break the while True loop.print("All inputs have been successfully downloaded.")break time.sleep(2)
Create a custom model
Now we can create a custom model that's going to be using the concepts positive and negative. All inputs (in our application) associated with these two concepts will be used as a training data, once we actually train the model.
Each model training produces a new model version. See on the bottom of the code example, that we put the model version ID into its own variable. We'll be using it below to specify which specific model version we want to use (since a model can have multiple versions).
whileTrue: get_model_response = stub.GetModel( service_pb2.GetModelRequest(model_id="my-text-model"), metadata=api_key_metadata )if get_model_response.status.code != status_code_pb2.SUCCESS:raiseException("Failed response, status: "+str(get_model_response.status)) version_status_code = get_model_response.model.model_version.status.codeif version_status_code == status_code_pb2.MODEL_TRAINED:print("The model has been successfully trained.")breakelif version_status_code in (status_code_pb2.MODEL_QUEUED_FOR_TRAINING, status_code_pb2.MODEL_TRAINING):print("The model hasn't been trained yet. Trying again shortly.") time.sleep(2)else: error_message = (str(get_model_response.status.code)+" "+ get_model_response.status.description +" "+ get_model_response.status.details )raiseException(f"Expected model to train, but got {error_message}. Full response: {get_model_response}" )model_version_id = get_model_response.model.model_version.id
Predict on new inputs
Now we can use the new custom model to predict new text examples.
post_model_outputs_response = stub.PostModelOutputs( service_pb2.PostModelOutputsRequest( model_id="my-text-model",# By default, the latest model version will be used, but it doesn't hurt to set it explicitly. version_id=model_version_id, inputs=[ resources_pb2.Input(data=resources_pb2.Data(text=resources_pb2.Text(raw="Butchart Gardens contains over 900 varieties of plants."))), resources_pb2.Input(data=resources_pb2.Data(text=resources_pb2.Text(url="https://samples.clarifai.com/negative_sentence_12.txt"))), ] ), metadata=api_key_metadata)if post_model_outputs_response.status.code != status_code_pb2.SUCCESS:raiseException("Failed response, status: "+str(post_model_outputs_response.status))for output in post_model_outputs_response.outputs: text_object = output.input.data.text val = text_object.raw if text_object.raw else text_object.urlprint(f"The following concepts were predicted for the input `{val}`:")for concept in output.data.concepts:print(f"\t{concept.name}: {concept.value:.2f}")
Start model evaluation
Let's now test the performance of the model by using model evaluation. See the the Model Evaluation page to learn more.
Model evaluation takes some time, depending on the amount of data in our model. Let's wait for it to complete, and print all the results that it gives us.
whileTrue: get_model_version_metrics_response = stub.GetModelVersionMetrics( service_pb2.GetModelVersionMetricsRequest( model_id="my-text-model", version_id=model_version_id, fields=resources_pb2.FieldsValue( confusion_matrix=True, cooccurrence_matrix=True, label_counts=True, binary_metrics=True, test_set=True, metrics_by_area=True, metrics_by_class=True, ) ), metadata=api_key_metadata )if get_model_version_metrics_response.status.code != status_code_pb2.SUCCESS:raiseException("Get model version metrics failed: "+str(get_model_version_metrics_response.status)) metrics_status_code = get_model_version_metrics_response.model_version.metrics.status.codeif metrics_status_code == status_code_pb2.MODEL_EVALUATED:print("The model has been successfully evaluated.")breakelif metrics_status_code in (status_code_pb2.MODEL_NOT_EVALUATED, status_code_pb2.MODEL_QUEUED_FOR_EVALUATION, status_code_pb2.MODEL_EVALUATING):print("The model hasn't been evaluated yet. Trying again shortly.") time.sleep(2)else: error_message = (str(get_model_version_metrics_response.status.code)+" "+ get_model_version_metrics_response.status.description +" "+ get_model_version_metrics_response.status.details )raiseException(f"Expected model to evaluate, but got {error_message}. Full response: {get_model_version_metrics_response}" )print("The model metrics response object:")print(get_model_version_metrics_response)