Is pg_trigger_depth() bad to use for preventing trigger cascading (recursion)?

It is not bad to use per se (IMHO). You just need to be aware of implications.

I'd rather make it pg_trigger_depth() < 1 instead of pg_trigger_depth() = 0, on principal, but that's not important here. We are talking about triggers like:

CREATE TRIGGER my_trigger
AFTER INSERT ON my_tbl
FOR EACH ROW
WHEN (pg_trigger_depth() < 1)
EXECUTE PROCEDURE my_trigger_func();

If you later add triggers that would in turn fire your trigger, nothing will happen instead. Your trigger only reacts to non-trigger events. Depending on your setup this may be exactly what you want - or kind of a trap, if you forget about this and later add more triggers that should in turn (also) fire this trigger. You are changing default behavior. Be sure to document this clearly.

Be aware that pg_trigger_depth() counts any trigger, not only the same trigger to prevent recursion, so the condition prevents the trigger from firing if called from any other trigger, too.
The manual:

current nesting level of PostgreSQL triggers (0 if not called, directly or indirectly, from inside a trigger)

Related:

  • How to prevent a PostgreSQL trigger from being fired by another trigger?

Yes, it's always bad to make behavior dependent on pg_trigger_depth()

Maybe I'm a little less averse to blanket statements, but what good could it do? There is no argument I can see as to why you would want such a feature. The primary purpose of a database is to ensure data integrity. And as far as I can see, and I may be wrong, such a use always is a potential violation of data-integrity. In @Erwin's answer he says "if you forget". The point of ensuring integrity is to eliminate that possibility. If an agent can remember everything and understand the consequences and code around them, you can get data integrity out of anything.

Let's introduce a few terms, in programming, we have

  • "state" which includes anything a programmer has access to.
  • "execution context" which includes the environment for execution.

We further have a term for a function, that has no state we call that a pure function,

The function always evaluates the same result value given the same argument value(s). The function result value cannot depend on any hidden information or state that may change while program execution proceeds or between different executions of the program, nor can it depend on any external input from I/O devices (usually—see below).

The distinction for purity is useful because it eliminates any burden to remember anything on behalf of the computer, or the programmer: f(x) = y is always true. Here you're violating purity in the worst place -- the database. And, you're doing it in a fashion that is complex and error prone -- making internal execution context a palpable thing in your DB application's state.

I wouldn't want that. Just consider the complexities you have to buffer mentally.

  • Table A 's Trigger A fires
    • Trigger A always issues DML to Table B, firing Trigger B.
      • Trigger B conditionally issues DML to Table A, firing Trigger A.
      • Trigger B always issues DML to Table A, firing Trigger A.
    • Trigger A conditionally issues DML to Table B, firing Trigger B.
      • Trigger B conditionally issues DML to Table A, firing Trigger A.
      • Trigger B always issues DML to Table A, firing Trigger A.

If that looks complex, keep in mind that "conditionally" there can be further expanded to "it happened, but it may not always happen", and "it didn't happen, but it may happen elsewhere."

For pg_trigger_depth() to even be useful, you have to have a series of events that is similarly complex. And, now, you want execution to be dependent on execution content in that chain?

I would avoid this. If your trigger system is this complex, you're playing hot potato with a hand grenade in a closet all by yourself and it doesn't seem likely to end well.